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本 书 是 一 本 与 众 不 同 的 SQL Server 数据 库 入 门 读 物 ， 不 需要 读者 有 太 多 基础 。 本 书 以 简单 实用 为 原 
则 ， 讲 解 通俗 易 懂 ， 循 序 渐进 ， 避 免 了 云 山 雾 曾 、 上 星 涩 难 懂 。 本 书 风格 轻松 活泼 ， 多 用 对 比 、 类 比 和 比 
喻 等 写作 方式 ， 并 配合 图 解 教学 ， 对 难点 之 处 给 出 了 必要 提示 ; 书 中 的 每 个 知识 点 都 对 应 相应 的 示例 ， 
便于 读者 一 边 学 习 一 边 动手 实践 ， 既 可 以 提高 动手 能 力 ， 也 可 以 激发 学 习 兴 趣 。 另 外 ， 本 书 配 工 张 光 盘 ， 
内 容 为 本 书 配 套 多 媒体 教学 视频 及 源 代码 。 本 书 还 赠送 了 400 多 个 SQL Server 实例 源 代码 及 12 小 时 教 
学 视频 ( 需 下 载 )。 

本 书 共 18 章 ， 分 为 5 篇。 第 1 篇 介绍 了 SQL Server 基础 知识 ， 包 括 SQL Server 数据 库 的 安装 、 数 
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篇 介绍 了 SQL Server 数据 库 常 用 的 一 些 对 象 ， 包 括 视 图 、 存 储 过 程 及 触发 器 等 对 象 ， 第 4 篇 介绍 了 SQL 
Server 数据 库 的 管理 ,包括 数据 库 的 备份 和 还 原 、 数 据 库 的 权限 管理 以 及 自动 化 任务 管理 ; 第 5 篇 介绍 了 
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SQL Server 数据 库 是 目前 比较 流行 的 数据 库 之 一 ， 与 其 他 数据 库 产品 一 样 ， 都 可 以 使 
标准 的 SQL 语句 。SQL Server 数据 库 赁 借 其 自身 的 操作 简单 ， 与 Windows 操作 系统 的 
融合 性 以 及 与 Visual Studio 开发 平台 的 集成 性 ， 深 受用 户 的 喜爱 。 目 前 ， 在 很 多 的 中 小 型 
网 站 和 软件 系统 中 都 普遍 应 用 SQL Server 作为 后 台数 据 库 。 

为 了 能 够 让 读者 快速 掌握 SQL Server 的 使 用 ， 笔 者 编写 了 本 书 。 本 书 从 SQL Server 
数据 库 的 安装 开始 讲 起 , 循序 渐进 地 介绍 了 SQL Server 数 据 库 操作 和 管理 的 方方面面 知识 。 
从 基本 概念 到 具体 实践 ， 从 新 特性 的 讲解 到 具体 操作 ， 从 简单 的 SQL 语句 编写 到 复杂 的 数 
据 库 管理 ， 从 抽象 概念 到 实际 应 用 ， 全 方位 解读 了 SQL Server 数据 库 的 相关 知识 。 本 书 最 
后 两 章 介 绍 了 如 何 使 用 目前 比较 主流 的 C# 和 Java 语言 在 编程 中 连接 SQL Server 数据 库 ， 
可 以 让 读者 对 实际 的 数据 库 应 用 开发 有 一 个 直观 的 了 解 。 

本 书 将 知识 范围 锁定 在 了 适合 初级 和 中 级 读者 阅读 的 部 分 ， 讲 解 时 结合 了 大 量 示例 ， 
专门 录制 了 多 媒体 教学 视频 辅助 教学 。 相 信 读 者 通过 学 习 本 书 内 容 ， 可 以 比较 好 地 掌握 
SQL Server 数据 库 的 相关 知识 ， 为 自己 的 IT 职业 生涯 做 好 准备 。 


本 书 特色 


本 书 奉 行 “ 入 门 很 简单 丛书” 的 一 贯 风 格 ， 有 以 下 突出 特色 : 

专门 提供 配套 多 媒体 教学 视频 ， 便 于 读者 更 加 直观 、 高 效 地 学 习 ， 增 强 学 习 效果 。 
编排 采用 循序 渐进 的 方式 ,适合 初 、 中 级 学 者 快速 掌握 SQL Server 数据 库 的 使 用 。 
采用 语法 与 示例 一 对 一 的 方式 来 讲解 每 一 个 语法 ， 可 以 让 读者 更 加 牢固 地 掌握 。 
结合 大 量 实例 讲解 SQL Server 中 的 基本 SQL 语句 和 企业 管理 器 的 使 用 。 

所 有 实例 都 具有 代表 性 和 实际 意义 ， 能 够 解决 工作 中 的 实际 问题 。 

对 于 在 SQL Server 中 编写 语句 比较 容易 出 现 的 问题 ， 给 出 了 详细 的 说 明 。 
提供 了 利用 C# 和 Java 语言 连接 SQL Server 数据 库 的 案例 ， 可 以 帮助 读者 体会 实 
际 开 发 中 SQL Server 数据 库 的 使 用 。 

口 本 书 提供 了 大 量 练习 题 ， 以 帮助 读者 巩固 和 提高 所 学 的 知识 。 


本 书 的 内 容 安排 
本 书 共 18 章 ， 分 为 以 下 5 篇 。 
第 1 篇 走 进 SQL Server (第 1~3 章 ) 


本 篇 首先 介绍 了 SQL Server 数据 库 在 Windows 环境 下 的 安装 过 程 及 每 个 数据 库 版 本 
的 说 明 ， 然 后 介绍 了 数据 库 的 创建 、 修 改 及 删除 ， 以 及 创建 数据 表 、 修 改 及 删除 数据 表 等 。 


DOOOODODO 


SQL Server 入 门 很 简单 


第 2 篇 。 表 操作 基础 (第 4~8 章 ) 


本 篇 主要 介绍 了 如 何 使 用 表 中 约束 ， 以 及 如 何 操作 表 中 的 数据 、 如 何 使 用 函数 等 。 主 
要 包括 数据 表 中 数据 的 添加 、 修 改 及 删除 ， 数 据 表 中 数据 的 简单 查询 和 复杂 查询 ， 在 查询 
语句 中 使 用 函数 来 方便 数据 查询 。 


第 3 篇 ”数据 库 使 用 进 阶 (第 9~13 章 ) 


有 了 数据 表 操作 的 基础 后 , 就 可 以 灵活 地 使 用 SQL 语句 来 更 好 地 使 用 数据 库 。 本 篇 主 
要 介绍 了 SQL Server 中 视图 、 索 引 、 存 储 过 程 及 触发 器 的 使 用 。 


第 4 篇 数据 库 的 管理 (第 14~16 章 ) 


有 了 前 3 篇 的 基础 后 ， 已 经 对 数据 库 的 基本 操作 有 所 了 解 。 在 本 篇 中 主要 介绍 了 数据 
库 的 管理 知识 ， 包 括 数据 库 的 备份 和 还 原 、 用 户 和 权限 管理 及 系统 化 自动 任务 管理 。 


第 5 篇 ”数据 库 的 应 用 (第 17~18 章 ) 


本 篇 介绍 分 别 使 用 C# 语 言 和 Java 语言 连接 SQL Server 数据 库 的 相关 知识 。 在 使 用 C# 
语言 连接 数据 库 部 分 ， 以 文章 管理 系统 为 例 让 读者 更 加 熟悉 SQL Server 数据 库 的 使 用 ; 在 
使 用 Java 语言 连接 数据 库 部 分 , 介绍 了 如 何 使 用 Java 语言 连接 SQL Server 完成 订购 系统 。 


适合 阅读 本 书 的 读者 


口 从 未 接触 过 SQL Server 的 自学 人 员 ; 
口 打算 使 用 SQL Server 数据 库 的 开发 人 员 ; 
口 大 中 专 院 校 的 学 生 和 相关 授课 老师 ; 
口 准备 从 事 软 件 开发 的 求职 者 ; 
口 参与 毕业 设计 的 学 生 ; 
口 其 他 编程 爱好 者 。 
本 书 作者 


本 书 由 秦 婧 主笔 编写 。 其 他 参与 编写 的 人 员 有 了 丁 士 锋 、 胡 可 、 姜 永 艳 、 新 鳃 鹏 、 孔 峰 、 
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郑 伟 、 郑 玉 晖 、 朱 雪琴 、 戴 思 齐 、 丁 航 峰 。 

阅读 本 书 的 过 程 中 若 有 疑问 ,请 发 邮件 和 我 们 联系 。E-mail:bookservice2008@163.com。 
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第 1 章 初 识 数据 库 


在 学 习 任 何事 物 之 前 ， 都 是 要 先 了 解 这 个 事物 是 什么 。 当 然 ， 学 习 数据 库 也 不 例外 ， 

要 首先 知道 数据 库 能 够 做 什么 以 及 如 何 安装 数据 库 ， 才 能 够 有 针对 性 地 根据 自己 的 需要 来 
学 习 。 在 本 章 中 ， 将 带领 读者 全 面 了 解数 据 库 中 的 一 些 主流 产品 ， 并 着 重 讲解 SQL Server 
章 的 主要 知识 点 如 下 : 

数据 库 和 数据 库 系统 

常用 的 数据 库 产 品 

安装 SQL Server 数据 库 

SQL Server 工作 平台 介绍 

SQL Server 中 自 带 的 数据 库 


1.1 与 数据 库 有 关 的 一 些 概 念 


提 到 数据 库 就 会 出 现 一 系列 的 概念 ， 比 如 数据 库 系 统 和 数据 库 管 理 系 统 。 只 有 理解 好 
这 些 与 数据 库 相 关 的 概念 ， 才 能 够 更 好 地 掌握 数据 库 。 在 本 小 节 中 ， 将 简单 地 为 读者 解释 
这 些 需要 知道 的 数据 库 概念 。 
1.1.1 数据 库 

数据 库 (Database) 简称 DB， 是 指 用 来 存放 数据 的 仓库 。 这 就 好 像 是 超市 用 来 存放 商 
品 的 仓库 一 样 ， 在 仓库 中 有 着 各 种 各 样 的 商品 。 在 数据 库 中 存放 的 就 不 是 商品 了 ， 只 能 是 
数据 。 但 是 ， 这 些 数据 并 不 一 定 就 是 数字 ， 也 可 以 是 文字 、 图 片 或 者 一 段 视频 等 信息 。 当 
把 数据 存放 在 数据 库 中 ， 数 据 就 可 以 长 期 地 存放 了 。 
1.1.2 ”数据 库 管 理 系 统 

数据 库 管 理 系 统 (Database Management System ) 简称 DBMS， 是 指 用 来 管理 数据 库 的 
一 种 软件 。 通 过 数据 库 管理 系统 可 以 方便 地 对 数据 库 中 的 数据 进行 操作 。 数 据 库 管理 通常 
会 有 数据 定义 功能 、 数 据 操作 功能 以 及 维护 数据 库 安 全 的 功能 。 实 际 上 ， 数 据 库 管理 系统 
中 使 用 的 语言 就 是 SQL 语言 ， 即 结构 化 查询 语言 。 本 书 要 学 习 的 数据 库 SQL Server 也 是 
通过 SQL 语言 来 操作 的 。 
1.1.3 ”数据库 系 统 

数据 库 系统 (Database System) 简称 DBS， 是 指 在 系统 中 使 用 了 数据 库 管理 数据 。 通 
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常数 据 库 系统 包含 了 数据 库 、 数 据 库 管 理 系统 、 用 户 以 及 操作 系统 、 计 算 机 硬件 等 元 素 。 
换 名 话说， 数据 库 系统 就 是 前 面 所 讲 过 的 数据 库 和 数据 库 管 理 系统 的 综合 体 。 目 前 ， 在 企 
业 中 ,广泛 地 使 用 着 数据 库 系统 ， 比 如 : 办 公 自 动 化 系统 、 电 子 考勤 系统 等 等 。 


1.2 了 解 常用 的 数据 库 产品 


在 本 书 中 要 学 习 的 是 SQL Server 数据 库 ， 但 并 不 是 说 目前 只 有 这 一 款 数据 库 产 品 。 目 
前 ， 在 企业 和 个 人 所 使 用 的 数据 库 产 品 中 , 有 几 款 产品 的 市 场 占有 率 是 名 列 前 茅 的 ， 比 如 : 
Oracle 数据 库 、MySQL 数据 库 、Access 数据 库 以 及 本 书 中 要 学 习 的 SQL Server 数据 库 等 。 
通过 对 这 些 常用 的 数据 库 产 品 进行 了 解 ， 才 能 够 使 用 户 在 实际 应 用 中 更 游刃有余 地 选择 数 
据 库 产品 。 下 面 就 对 上 面 所 提 到 的 4 款 数 据 库 产品 做 个 简单 的 介绍 。 


1.2.1 _ Oracle 数据 库 


Oracle 数据 库 是 甲骨 文公 司 开 发 的 一 款 数据 库 产 品 ， 主 要 应 用 于 大 中 型 企业 。 目 前 ， 
主流 版 本 是 Oracle 11g。Oracle 数据 库 最 大 的 好 处 就 是 其 跨 平 台 的 特点 ， 能 够 满足 不 同 操 
作 系 统 的 使 用 。 同 时 ，Oracle 数据 库 以 其 自身 良好 的 安全 性 和 数据 存储 能 力 满足 了 大 型 企 
业 的 要 求 。 但 是 ， 由 于 Oracle 数据 库 并 不 是 免费 提供 的 ， 并 且 价格 并 不 便宜 ， 因 此 ， 对 于 
一 些 中 小 型 企业 还 是 望而却步 的 。 下 面 给 读者 见识 见识 Oracle 数据 库 的 企业 管理 器 界面 ， 
如 图 1.1 所 示 。 
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一 般 信息 主机 CPU 活动 会 话 数 SQL 响应 时 间 
a 人 i 和 6 
状态 应 寺 天 日 入 
开始 运行 时 间 2013-3-27 下 午 08 时 56 分 13 秒 csT so ; 让 用 05 
实例 名 orcl 本 [各 
本 cl 这 00 
a 站 00 国志 
asiaala 最 新 收 
加 。 页 活 cfu 最 大 值 1 =QL 响应 时 间 正 
可 后 
空间 概要 高 可 用 性 
数据 库 大 小 GE) 1552 实例 恢复 时 间 合 ) 亏 
有 问题 的 表 宁 间 。 0 上 次 备份 n/a 
段 指导 建 可 用 快速 恢复 区 百分比 100 
内 回 数 据 库 事件 记录 。 莹 用 


违反 策略 v 
已 用 转 储 区 百分比 (%) 


图 1.1 Oracle 11g 企业 管理 器 主 界 面 


1.2.2 MySQL 数据 库 


MySQL 数据 库 是 一 款 开源 数据 库 ， 目 前 常用 的 版 本 是 MySQL 5.5， 并 提供 免费 版 为 
户 使 用 。MySQL 数据 库 也 是 可 以 跨 平台 使 用 的 ， 通 常 被 中 小 型 企业 所 青睐 ， 尤 其 是 企 
业 在 选择 使 用 PHP 语言 作为 开发 语言 时 ， 一 般 首选 都 是 使 用 MySQL 数据 库 。 对 于 其 他 编 
程 语言 来 说 ， 选 择 MySQL 数据 库 的 并 不 是 太 多 。 安 装 后 的 MySQL 数据 库 是 在 DOS 界面 


。3。 
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下 打开 的 ， 如 图 1.2 所 示 。 


™ MySQL 5.5 Command Line Clieni 


Pee 
lelcone to the MySQL nonitor. Connands end with ; or \g- 

Nour MySQL connection id is 

erver version: 5.5.17 MySQL Conmunity Server 《GPL》 

opyright Cc> 2888. 2811, Oracle and/or its affiliates. All rights reserved. 
Oracle is a registered trademark of Oracle Corporation and/or its 
ffiliates. Other nanes nay be tradenarks of their respective 

puners. 


ype ’help;’ or ’\h’ for help. Type ’\c’ to clear the current input statement- 


ysql> 


图 1.2 MySQL 登录 界面 


1.2.3 Access 数据 库 


Access 数据 库 ， 读 者 应 该 不 陌生 ， 在 操作 系统 上 安装 了 Office 办 公 软 件 后 ， 就 会 出 现 
在 安装 的 列表 上 了 。 没 错 ，Access 数据 库 就 是 Office 办 公 软 件 中 的 一 个 软件 产品 。 它 与 
Office 同步 被 升级 ， 目 前 常用 的 版 本 是 Access 2007。 它 的 特点 是 使 用 方便 ， 在 安装 Office 
之 后 就 可 以 直接 使 用 了 ， 通 常 被 一 些小 型 企业 作为 开发 门户 网 站 的 首选 产品 。 但 是 ， 由 于 
Access 数据 库 只 能 在 Windows 系列 的 操作 系统 上 使 用 并 且 存 储 数据 的 容量 不 大 ,因此 , 它 
并 没有 被 广泛 地 应 用 。Access 数据 库 操作 界面 如 图 1.3 所 示 。 这 里 是 Access 的 2003 版 本 
的 数据 库 。 


scrosoft Access 
文件 中转 氏 下) 视 轩 插入 C) 工具 中 宣 口 ) 帮助 D 
BR- EAE YT WE ME ME AE IE MEY 


| 
国人 用 向 S 提 娃 表 
加 通 过 输入 数据 自 娃 表 


图 1.3 Access 数据 库 操作 界面 


1.2.4 SQL Server 数据 库 


SQL Server 数据 库 也 是 微软 公司 的 一 款 数据 库 产品 ， 目 前 常用 的 版 本 是 SQL Server 
2008， 也 是 本 书 中 要 学 习 的 版 本 。 该 数据 库 的 缺点 就 是 不 能 跨 平台 使 用 ， 但 是 它 在 其 他 方 
面 的 性 能 并 不 比 其 他 数据 库 逊 色 。SQL Server 数据 库 是 目前 大 中 型 企业 作为 软件 开发 时 选 
择 比较 多 的 一 款 产 品 ， 尤 其 是 在 应 用 微软 的 开发 平台 Visual Studio 来 开发 软件 时 。 此 外 ， 


.4°. 
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SQL Server 数据 库 提供 的 企业 管理 器 能 够 为 用 户 操 作 数 据 库 提供 方便 。 对 于 SQL Server 数 
据 库 ， 将 在 下 一 节 详 细 讲 解 其 安装 和 登录 的 过 程 。 


1.3 安装 SQL Server 2008 


SQL Server 2008 能 够 更 好 地 融合 其 他 版 本 的 优点 , 并 且 还 向 商业 智能 方面 迈 出 了 可 喜 
的 一 步 。 既 然 有 这 样 好 的 一 款 产 品 ， 相 信 读 者 已 经 迫不及待 地 想 学 习 了 。 但 是 ， 还 不 能 着 
急 ， 安 装 SQL Server 2008 才 是 学 好 它 的 第 一 步 。 


1.3.1 SQL Server 2008 各 版 本 介绍 


在 安装 SQL Server 2008 之 前 , 还 要 知道 在 什么 情况 下 使 用 什么 版 本 。 这 就 好 像 是 计算 
机 的 操作 系统 一 样 ， 当 只 是 为 了 在 家 上 网 使 用 ， 可 以 直接 安装 Windows 系列 的 家 庭 版 ; 当 
需要 做 软件 开发 使 用 ， 可 以 安装 Windows 的 服务 器 版 或 企业 版 。 因 此 ， 了 解 了 SQL Server 
2008 的 版 本 ， 对 使 用 它 还 是 有 帮助 的 。 目 前 ，SQL Server 2008 常用 的 版 本 主要 有 企业 版 、 
标准 版 、 工 作 组 版 、 简 易 版 以 及 开发 版 。 具 体 的 介绍 如 表 1-1 所 示 。 


表 1-1 SQL Server 2008 各 版 本 介绍 
版 本 说 明 
支持 32 位 或 64 位 系统 。 它 是 一 种 综合 的 数据 平台 ， 能 够 对 复杂 数据 进行 分 
企业 版 ‘Enterprise) 析 ， 拥 有 商业 智能 和 分 析 能 力 。 目 前 ， 可 以 免费 使 用 的 版 本 是 Evaluation 版 ， 
可 供 免 费 使 用 180 天 


标准 版 (Standard) 支持 32 位 或 64 位 系统 。 它 适合 中 小 型 企业 选用 ， 也 包括 企业 版 的 全 部 功能 
开发 版 (D 与 企业 版 的 功能 类 似 ， 只 是 不 能 够 作为 数据 库 服 务 器 使 用 。 仅 能 用 做 学 习 和 
eveloper ) 


测试 
工作 组 (Workgroup) Ee 32 位 操作 系统 。 它 是 一 款 入 门 级 的 数据 库 产 品 , 通常 适用 在 小 型 企业 
简易 版 (Express) 只 支持 32 位 操作 系统 。 它 是 一 款 免 费 的 软件 ， 是 Visual Studio 中 集成 的 产品 


除了 在 上 表 中 列 出 的 数据 库 产品 的 版 本 外 ， 还 有 一 款 专门 用 于 移动 设备 开发 和 使 用 的 
Compact 3.5 SP1(x86) 版 ， 它 也 是 免费 提供 。 在 本 书 中 所 使 用 的 是 SQL Server 2008 的 企业 
评估 版 “Enterprise Evaluation)， 该 版 本 可 以 在 微软 的 官方 网 站 上 免费 下 载 ， 并 可 以 免费 使 
用 180 天 。 


1.3.2 在 Windows Server 环境 下 安装 SQL Server 2008 


如 果 读 者 已 经 在 微软 的 官方 网 站 上 下 载 了 SQL Server 2008 的 企业 评估 版 , 也 不 要 着 急 
安装 。 先 要 确认 准备 安装 该 软件 的 操作 系统 是 否 支 持 该 软件 以 及 是 否 有 足够 的 容量 。 

首先 ， 如 果 您 的 操作 系统 是 Windows 系列 的 操作 系统 ， 那 么 就 可 以 松 一 口气 了 ， 没 问 
题 ， 可 以 安装 。 然 后 ， 在 操作 系统 支持 的 前 提 下 ， 再 看 看 硬盘 空间 是 否 足够 大 呢 ? 在 安装 
SQL Server 2008 的 过 程 中 ， 软 件 是 需要 至 少 2GB 的 磁盘 空间 来 存储 临时 文件 和 安装 后 的 
文件 。 如 果 对 自己 的 安装 环境 还 不 够 放心 的 话 ， 可 以 参考 微软 网 站 上 给 出 的 具体 要 求 
(http://msdn.microsoft.com/library/ms143506(SQL.100).aspx )。 
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好 了 , 准备 工作 已 经 完成 了 , 现在 就 要 开始 安装 了 。 本 例 是 将 该 数据 库 安装 到 Windows 
XP 下 。 安 装 SQL Server 2008 主要 分 为 如 下 几 个 步骤 。 


1. 打开 SQL Server 安装 中 心 界面 


打开 下 载 后 的 安装 文件 目录 ,双击 setup.exe 可 执行 文件 , 然后 会 弹出 需要 安装 Windows 
Install 和 .NET Framework 框架 的 提示 。 如 果 读 者 的 计算 机 中 没有 上 述 两 个 软件 ， 还 需要 从 
微软 的 官方 网 站 上 下 载 并 安装 。 如 果 已 经 安装 了 这 两 个 软件 ， 那 么 ， 单 击 “确定 ”按钮 ， 
即 可 打开 安装 向 导 界 面 ， 如 图 1.4 所 示 。 


ER 
二 seman. 
EB <2 文 
到 在 安全 文档 
EB EL 人 行 % 时 
OB ssn. 


a 
计划 
EE 
维护 
IR 
要 更 
所 
地 项 


全 [serverzxa 


,这 下 二 和 要 地 吾 吕 

Beran se 
ES 
逢 顾问 分 林 安 新 的 所 育 5ot serwer 2005 或 SQL Servel 2000 相生 ， 并 在 升 到 SQL 5erver 
2008 之 前 于 之 后 明定 要 六 的 问题 

国生 了 机 村 和 内 4 

局 到 HB 轩 说 
| 各、 加 何 开 培 借用 SQL Server 2006 站 隆 特 称 幸 入 
二 <Teosragn se se 2000 term. 


EE 升 全 文 肖 
查看 有 关 加 何 作 SQ server 2000 王 SQL Server 2005 升 全 到 SQL Server 2008 的 廊 档 。 


图 1.4 SQL Server 安装 中 心 界 面 


2. 选择 安装 选项 
在 图 1.4 所 示 的 界面 中 ， 单 击 “ 安 装 ” 选 项 ， 出 现 图 1.5 所 示 的 界面 。 


计划 全 了 全 条 3 5erver 入 立 安装 或 向 现 有 安装 逢 加 功 密 
安装 “起 启动 向 导 ， 以 在 村 群集 环境 中 安装 SQL Server 2003 ， 或 者 向 钢 有 SQL Server 2008 实例 中 添 


汪 52 server 8 用 特 你 人 安 守 
i “各 忆 动 向 导 以 半音 节 上 的 5QL Sever 2006 前 隆 斩获 闫 全， 
i 
把 ， 避 全 SQ server 2008 隐 壬 和 和 表 中 基 加 节 训 。 
进项 
看 从 5Qt serverz000 或 5QL 5errer 0 天 级 
但 局 动身 导 忆 村 5QL server an00 或 5QL 3erver Zt05 升 弛 到 SQL server Z009， 在 升级 朋 ,应 先 
运行 升 似 硕 月 ， 以 检测 可 对 存在 的 请 是. 
已 
搜索 Merecoh Updale 上 的 SQL serve 2008 产品 更 新 - 


图 1.5 选择 安装 选项 界面 
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在 这 里 ， 单 击 第 一 个 选项 “全 新 SQL Server 独立 安装 或 向 现 有 安装 添加 功能 ” 开始 
安装 。 首 先是 要 确认 安装 程序 是 否 能 安装 到 当前 的 环境 中 ， 如 图 1.6 所 示 。 


辐 区 FRI 
刁 过 srve 六 伯 上 文件 的 & 由 各 


图 1.6 安装 程序 支持 规则 


只 要 读者 的 安装 效果 与 图 1.6 所 示 一 样 ， 都 是 通过 状态 。 那 么 ， 就 可 以 单 击 “ 确 定 ” 
按钮 进一步 安装 SQL Server 了 。 


3. 输入 SQL Server 的 安装 密 钥 
完成 了 图 1.6 所 示 的 操作 后 ， 进 入 输入 安装 SQL Server 密 钥 的 界面 ， 如 图 1.7 所 示 。 


镑 5QL Server 2008 安装 程序 有 DG 可 
产品 密 钥 
指定 要 安装 的 5QL 5erver 2009 版 本 。 

产品 灾 铀 二 ever 0 ot 

许可 条 区 下 各 证 字 王 各 包装 上 的 25 个 字符 的 省 角 各 来 作 指 定 Eterprke EvauaNbn ， 则 该 人 阿 在 诉 笑 后 村 
EE 

安装 程序 支持 文件 
个 指定 可 用 版 本 (5): 

Enterprise Evauaton 了 

个 输入 产品 外 人; 


S20 
图 1.7 输入 产品 密 钥 界 面 
这 里 ， 选 择 指定 的 安装 版 本 Enterprise Evaluation， 不 必 输 入 产品 密 钥 。 


BE 


。T。 


第 1 篇 走 进 SQL Server 


4. 阅读 SQL Server 的 许可 条 款 


在 图 1.7 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 阅读 许可 条 款 界面 ， 如 图 1.8 
所 示 。 


MICROSOFT 评 估 软 件 许可 条 款 


MICROSOFT SQL SERVER 2008 EVALUATION EDITION 
Microsof Corporation (或 你 所 在 地 的 Microso 人 tCorporaton 关联 公司 ) 与 您 之 间 
A 
。 更 新 
。 i. 
。 ”基于 Inernet 的 服务 和 
。 支持 服务 
果 确 实 附带 有 其 他 条 款 ， 则 其 他 冬训 应 适用 。 
旦 使 用 该 软件 ， 则 表明 念 接受 这 此 条款。 如 果 您 不 接受 这 些 条 款 ， 请 不 要 使 用 该 软件 。 


果 你 名 守 这 些许 可 条 蒜 ， 念 将 具有 下 列 权利 。 Ld 
si Ro 
本 琅 有 于 可 条 浆 (*] 
EE 


图 1.8 阅读 许可 条 款 界面 
在 阅读 完 条 款 后 ， 选 中 “我 接受 许可 条 款 ” 复 选 框 。 
5. 安装 程序 支持 文件 
在 图 1.8 所 示 的 界面 中 ， 单 击 “ 下 一 步 ”按钮 ， 即 可 进入 安装 程序 支持 文件 界面 ， 如 


图 1.9 所 示 。 


单 安 装 “以 安装 安装 程序 到 失 文 件 。 基 要 安装 或 更 新 SQL Server 2008， 这 些 女 件 是 必需 的 。 


产品 家 Sal emer 安 半 程序 需要 区 央 件 (D 


许可 条款 5 3 
安 半 程序 过 桂 立 件 
E22 | 


图 1.9 安装 程序 支持 文件 界面 


单 击 “ 安 装 ” 按 钮 ， 即 可 玫 
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始 安装 程序 ， 效 果 如 图 1.10 所 示 。 


< 上 | Ts >| mw 到 
图 1.10 安装 程序 支持 的 文件 


| 


在 图 1.10 所 示 的 界面 中 ， 列 出 了 安装 程序 支持 的 文件 ， 如 果 没 有 失败 的 状态 ， 就 说 明 


安装 程序 支持 文件 成 功 了 。 
6. 选择 需要 安装 的 功能 


在 图 1.10 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 安装 功能 选择 界面 ， 如 图 1.11 


镶 50L Server 2008 安装 程序 
功能 选择 


迁 择 要 安装 的 bnterprae Evalualion 功能 。 对 于 疼 集 实 装 ， 内 能 对 入 化 数 血 库 引 | 党 服务 和 Analysis Services 


DEE): 说 明 | 
ED 医 ERT 
ET 人 
回 sa Sever 复制 有 i: 
加 全文 搜索 
回 Anaysk Sevicss 


回 usess Irteliyence Developmert Sludo 
回 客 P 演 工具 连 按 


nie aten Servess 
固 客 尸 庙 工 具 向 后 车 窜 性 
加 PMIR ok 


回 Vicroscft SyncFranework 


可 再 发 行 的 功能 
全 部 下 
i 于 


< 上 -上 四 | 下 ->| MW | 和 


图 1.11 安装 功能 选择 界面 


这 里 ， 为 了 让 读者 能 够 全 面 学 习 SQL Server 2008， 选 择 “ 全 选 ”按钮 ， 将 所 有 功能 都 


选择 。 读 者 可 以 为 程序 选择 一 个 存放 目录 ， 笔 者 将 该 程序 安装 到 了 D 盘 下 。 


9。 
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7. 实例 配置 
在 图 1.11 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 实例 配置 界面 ， 如 图 1.12 所 示 。 


=lolxl 
实例 配置 
指定 5Qt Saver 实例 的 各 称 和 实例 ID 
安装 程 序 持 规则 6 MARND 
功 肖 入 反 全 名 实例 (: FEeee yy 
实例 本 办 
出 全 汪 要 下 
限 务 中 2 轩 RIOD: [a 
小 所 库 引 芝 配置 » 
| 
Reportng Senices 妈 二 
语调 和 使用 全 只 报 此 Sot Server Bl 录 : 由 Progyan FiesMiaoscft SQL ServerIM55QL10 MSsQLSERYER 
安装 规则 nalysis Services 目录: 由 rogram FieiMiaoerft SQL ServerWMSAS10 NGSGLSERVER 
Repcrting Sorvces 目录; diProy om ricsWMiaoscft SQL ServerIM5R510,MG5QLSCRVER 
闪 上 已 守 的 实 人 L: 


< 上 一 步 B)| 下 一步 > 职 渍 寻 助 上 


图 1.12 ”实例 配置 界面 
这 里 ， 使 用 默认 实例 ， 并 选择 实例 的 根 目 录 。 
8. 检测 硬盘 空间 
在 图 1.12 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 检测 硬盘 空间 的 界面 ， 如 图 1.13 


所 示 。 


[slol¥ 


功能 拓 ee TCD 

0 系统 纪 动 (cj: 需要 1256 MB 

磁 笃 空间 要 求 局 罗 38 可 0: 需要 1426 MB, 有 27773 MB 可 用 

服务 器 有 o 置 ri rene Fle ome seve 需要 734 MB 
i 实测 目录 diProoran FlesMiaoseft SQL enver): 需要 644 MB 
analyss Services 配置 

Reportng Servees 妈 置 

卉 误 和 使 用 恒 哆 报告 

安 半 坑 则 

众 各 安装 

安装 洲 讼 

EE 


nls ww | wm | 


图 1.13 检测 硬盘 空间 界面 
如 果 读 者 在 此 页 面 中 没有 发 现 未 通过 的 信息 ， 就 说 明 硬盘 的 容量 满足 需求 了 。 当 然 ， 
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一 般 情况 下 都 会 满足 需求 的 ， 毕 竞 现 在 的 硬盘 容量 都 很 大 了 。 
9. 服务 器 配置 
在 图 1.13 所 示 的 界面 中 ， 单 击 “ 下 一 步 ”按钮 ， 进 入 图 1.14 所 示 的 服务 器 配置 界面 。 


安装 程序 支持 规则 报 务 帐户 | 扫 序 规则 | 
功能 选择 
Tm Merosoft 建议 疙 对 每 个 SQL 5erver 服务 使 用 一 个 单独 的 眉 户 (MM)。 


出 全 要 下 
服务 加 配置 到 
小 所 库 引 芝 配置 一 
anayss Serviees 配置 司 
Reportng 5ervices 寻 置 副 
请 并 和 合用 情史 报 洛 

ET 习 
从 各 安 六 

安 音 进 度 对 了 有 5QL serve 服务 使用 相同 的 由 人 


将 在 可 能 的 情况 下 扎 动 加 轩 渤 些 职务 以 使 用 会 灶 可 由 户 。 在 一 些 软 旧 的 Windors 版 本 上 ,用户 需 


要 指定 低 特权 帐户 。 有 关 更 多 信息 ,请 皇 市 ` 帮 动人 了 


< 上 - 步 @)| 下 -— 步 (W) > 职 王 必 动 骨 
图 1.14 服务 器 配置 界面 
这 里 , 是 为 不 同 的 服务 设置 账户 名 和 密码 , 如 果 读 者 不 能 够 记 住 那么 多 的 账户 和 密码 ， 
建议 读者 选择 “对 所 有 SQL Server 服务 使 用 相同 账户 ”的 选项 。 笔 者 在 这 里 选择 的 就 是 该 
从 注意; 这 里 的 账户 指 的 是 当前 登录 Windows 系统 的 账户 。 如 果 当 前 登录 用 户 没有 登录 
密码 ， 则 需要 设置 密码 ， 这 个 密码 一 定 要 牢记 哦 ! 


10. 数据 库 引擎 配置 


在 图 1.14 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 数据 库 引 擎 配置 界面 ， 如 图 1.15 
在 此 界面 中 ， 用 来 配置 账户 的 身份 验证 模式 。 这 里 ， 选 择 “ 混 合 模式 〈SQL Server 
身份 验证 和 Windows 身份 验证 )” 的 选项 ， 并 在 下 面 为 内 置 的 SQL Server 管理 员 输 入 密 
码 。 在 下 面 通过 单 击 “ 添 加 当前 用 户 ” 按 钮 ， 指 定 当 前 登录 的 用 户 就 是 SQL server 的 管 


理 员 。 


11. Analysis Services 配置 


配置 好 用 户 信 息 后 , 单 击 “ 下 一 步 ” 按 钮 ,进入 Analysis Services 配置 界面 ， 如 图 1.16 
所 示 


在 此 界面 中 ， 单 击 “ 添 加 当前 用 户 ” 按 钮 ， 使 该 用 户 具 有 Analysis Services 服务 的 访 
问 权 限 。 
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SE 

数据 库 引擎 配置 
指定 数据 库 引 舍身 内 验证 安全 模式 、 管理 员 和 和 激 大昌 录 , 
A 本 | 有 | mesrrean | 
好 
人 为 后 诛 引 指定 身 全 证 术 式 和 半 理 员 . 
砸 县 到 司 要 求 二 协 验 证 模式 
人 个 wndows 身 从 验证 模式 全 ) 
配置 人 混合 要 式 (5QL Server 身价 验证 和 Windows 身份 验证 X 罗 
Reportng Services 辽 本 内 置 的 SQL 5ervert 系统 生理 风 帐 户 
境 误 和 使用 全 吕 孙 和 输入 这 码 昌 Fr 
众 各 六 确认 宕 史 (@): | 
安 半 进度 指定 5Qt Server 管理 只 
bn IBM-1402941575F\bm (i 
人 
权限 
EEC 
< 上 - 步 加 | 下 -上 由 >| 了 涌 | 
图 1.15 ”数据库 引擎 配置 界面 
=|olxl 


Analysis Services 配置 
指定 Andyae Servkes 管理 届 和 数 提 文人 严 。 


安装 程序 支持 规则 帐 P 设 置 | 尖 据 有 对 | 
2 指定 世上 用 户 具有 对 Analyss Servees 的 各 再 权限 。 


Reportng Services 卫 置 
错误 和 使 用 情况 报告 
安装 规则 


惟 备 安装 
安装 进度 
守成 


suo | mmo| 
< 上 | o> me | | 
图 1.16 ”Analysis Services 服务 配置 界面 


12. 配置 Reporting Services 


在 图 1.16 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 Reporting Services 配置 界面 。 如 
图 1.17 所 示 。 
这 里 ， 选 中 “安装 本 机 模式 默认 配置 ”选项 。 


13. 错误 和 使 用 报告 情况 


在 图 1.17 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 错误 和 使 用 报告 情况 界面 ， 如 
图 1.18 所 示 。 
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=Iolx] 
功能 选择 安装 本 机 可 式 对 认 配 村 [。 
实 创 配 雪 
栅 本 辣 要 求 安装 得 床 格 安 装 扒 表 服 务 器 并 正二 机 要 区 下 将 克 村 为 使 用 到 认证 。 安 疙 程序 充 吉 后 即 可 使 用 报 有 有 
服务 8 也 加 和 
数据 库 引 区 配置 
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ee 内 有 在 相生 hee 人， 后 台 拓 的 人 个， 并 有 到 所 
sharepomt 产品 或 法 地 的 实 列 上 实 半 并 本 要 了 用 于 ShrePoint 
礁 各 安 疙 后 ， 才 全 支 和 集成 六 作 。 
安装 进度 
Rm 安装 旧 丰 配置 有 表 服 务 器 (G。 


到 装 溃 尿 交 安装 但 不 卫生 扰 表 职务 器 软 首 。 安 装 元 艳 后 ， 烛 可 以 便 用 Reportng Services 卫 杜 工具 讼 村 
运行 好 表 服务 器 所 的 选项 


ts mw | ww | 
图 1.17 Reporting Services 配置 界面 
E69 
帮助 Mieresoft 改 加 3QL Server 功能 和 服务 。 
ee 和 
和 存 的 自动 更 新 “设置 ， 这 些 更 砷 可 能 会 自动 下 载 并 安装 汉 | 咎 的 计算 机 上 ， 
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村 国庆 Mosft Update 生动 36 的 本 人 让 
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Peporing Servker 恕 置 
Doshiseni Lo 厂 后 Windows 5ever 请 谈 孤 营 发 运 到 Moosdft 或 烙 公司 即 卫 洁 邮 务 响 。 设 设置 仅 造 用 于 以 无 用 
安装 规则 尸 区 互 方 3S(W)» 
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安装 进度 
污 成 


厂 斤 矶 能 信 月 情 况 涩 各 委 渤 到 Mecrocoft、 功 能 使 月 情况 数 疾 色 的 有 关 低 的 鲁 件 也 置 以 及 容 对 Merocoft 
Et te 


sm] mw | mw | 
图 1.18 错误 和 使 用 报告 情况 界面 
在 此 界面 中 ， 如 果 希 望 发 送 错误 报告 到 微软 ， 就 选中 复 选 框 ， 否 则 可 以 不 选 。 
14. 检查 安装 规则 


在 图 1.18 所 示 的 界面 中 ， 单 击 “ 下 一 步 ”按钮 ， 进 入 安装 规则 检查 界面 ， 如 图 1.19 
所 示 。 
这 里 ， 如 果 全 部 都 是 “已 通过 ”的 状态 ， 就 说 明 检测 成 功 了 。 


15. 检查 要 安装 项 


在 图 1.19 所 示 的 界面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 进 入 检查 要 安装 项 目的 界面 ， 如 
图 1.20 所 示 。 


旺 和 各 这 
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EE 
安装 规则 
安装 程序 正在 运行 观 则 以 确定 是 否 格 阻止 守 过程。 有 有 关 更 多 信息 ， 请 单 击 "和 助 ~. 


可 Ba 


加 | heparing servees 目 录 数 友 库 文件 存在 
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@ |sQ server 2005 Express I 具 


旧 昌 9 


< 上 -| 下 - 沙 () >| 取消 得 且 
图 1.19 安装 规则 检查 界面 


数据 库 引 尝 配置 数据 库 引 区 服务 
Anaysis Sevices 卫 置 5QL sever 复制 
Reporting 5ervices 配置 ee 
错误 和 使 用 情况 报 se 
安装 规则 Business Intelicence Development Studio 
礁 备 安装 喜 户 六 了 具 连 这 
安装 进 integration Servkes 
袍 记 人 和 己基 工具 癌 后 要 安生 
客户 油 T 具 Stk 
SQL server 联机 用 书 
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< 上 -四 | SO |] 9 种 且 L 
图 1.20 检查 要 安装 项 目 界 面 


16. 完成 安装 过 程 


在 图 1.20 所 示 的 界面 中 ， 单 击 “ 安 装 ” 按 钮 ， 即 可 开始 安装 SQL Server 软件 。 在 安装 
完成 后 ， 会 出 现 图 1.21 所 示 的 界面 。 
在 此 可 以 看 出 ， 所 有 的 功能 全 部 安装 成 功 了 。 


17. 确认 安装 


在 图 1.21 所 示 的 界面 中 ， 单 击 “ 下 一 步 ”按钮 ， 进 入 安装 成 功 界面 ， 如 图 1.22 所 示 。 
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图 1.21 安装 过 程 完成 界面 
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图 1.22 ”安装 成 功 界面 
在 此 界面 中 ， 可 以 看 到 已 经 将 日 志文 件 保 存 到 了 指定 位 置 。 单 击 “ 关 闭 ” 按 钮 ， 恭 喜 
你 ， 已 经 完成 了 SQL Server 2008 的 安装 。 
1.4 ”如 何 才 能 进入 SQL Server 


在 上 一 节 中 Es - Server 的 安装 , 那么 如 何 进入 到 SQL Server 数据 库 中 呢 ? 
记 住 ， 要 分 两 步 走 。 第 一 步 是 要 启动 SQL Server 的 数据 库 服务 ， 第 二 步 是 登录 SQL Server 
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1.4.1 启动 SQL Server 数据 库 服 务 


SQL Server 的 启动 实际 上 只 需要 启动 一 个 服务 即 可 ， 这 就 是 在 安装 数据 库 时 配置 的 实 
例 名 。 通 常会 有 两 种 方法 来 启动 SQL Server 的 数据 库 服务 : 一 种 是 在 管理 工具 中 的 服务 列 
表 中 启动 ;一 种 是 直接 使 用 SQL Server 安装 程序 后 从 自 带 的 SQL Server 配 置 管理 器 中 启动 。 
下 面 就 分 别 来 介绍 这 两 种 方法 。 

1. 在 管理 工具 中 的 服务 列表 中 启动 

单 击 “ 开 始 ” | “设置 ” |“ 控制 面板 ”选项 ， 在 出 现 的 控制 面板 列表 项 中 ， 单 击 “ 管 理 
工具 ”图 标 ， 并 在 打开 的 界面 中 单 击 “ 服 务 ” 图 标 ， 出 现 图 1.23 所 示 界 面 。 

aa 


文件 (E) ”操作 (4) ”查看 (VY) 才 助 (H) 
和 包 光 | 加 | 加 区 | 国 困 | ”于 中 本 


名 85QL Active Directory Helper 服务 
SQL FulHtext Fiter Daemon Launcher (MS55QLSERVER,，， 
5QL Server (M55QLSERVER2008) 


SQ Server (SQLEXPRESS) 
二 5QL Server Analysis Services (M55QLSERVER2008) 
乱 5QL Server Browser 
纺 5QL Server Integration Services 10.0 
二 5QL Server Reporting Services (M55QLSERVER2008) 
坊 y5QL Server V55 Writer 


雹 5QL server 代理 (M55QLSERYVER2008) 
纺 y5QL 5erver 代理 (5QLEXPRES5) 


ba 守 
[基站 站 证 闪闪 加 加 闪 加 如 如 加 加 闪避 


扩展 入 标准 / 


图 1.23 管理 工具 中 的 服务 列表 界面 


在 图 1.23 所 示 界 面 中 ， 用 黑 框 括 起 来 的 就 是 需要 启动 的 服务 SQL Server 
(MSSQLSERVER2008)， 其 中 MSSQLSERVER2008 就 是 在 安装 数据 库 时 设置 的 实例 名 。 从 
服务 的 状态 来 看 ， 目 前 该 服务 处 于 “已 启动 ”状态 。 如 果 该 服务 的 状态 处 什么 都 没有 写 ， 
就 说 明 没有 启动 该 服务 。 启 动 服务 非常 简单 ， 只 需要 右 击 该 服务 ， 在 弹出 的 右键 菜单 中 选 
择 “ 启 动 ” 选 项 ， 即 可 启动 该 服务 了 。 


2. 在 SQL Server 的 配置 管理 器 中 启动 服务 


在 安装 了 SQL Server 2008 数据 库 后 ， 就 会 在 
“开始 ”菜单 中 找到 SQL Server 的 配置 管理 器 。 具 


体 的 方法 就 是 , 单 击 “ 开 始 ”|“ 程 序 ”|Microsoft SQL - a 
Server 2008|“ 配 置 工具 ”|“SQL Server 配置 管理 器 ” me Bee a 
选项 ， 出 现 图 1.24 所 示 界 面 。 | 


在 图 1.24 所 示 界 面 中 , 单 击 “SQL Server 服务 ” : | 
选项 ， 出 现 图 1.25 所 示 的 SQL Server 服务 列表 。 图 1.24 SQL Server 配置 管理 器 界面 
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SQL Server Integration Services 10.0 ”正在 运行 
全 5QL Server Analysis 5ervices (M55,,， 正在 运行 
正在 运行 


图 1.25 SQL Server 配置 管理 器 中 的 服务 列表 界面 


在 图 1.25 所 示 界 面 中 , 列 出 的 就 是 SQL Server 中 所 使 用 的 全 部 服务 了 , 相信 读者 已 经 
找到 了 需要 启动 的 服务 ， 没 错 ， 就 是 SQL Server (MSSQLSERVER2008)。 启 动 的 方法 也 是 
右 击 该 服务 ， 在 弹出 的 右键 菜单 中 选择 “启动 ”选项 ， 即 可 启动 该 服务 了 。 


1.4.2 ”登录 SQL Server 数据 库 


既然 服务 已 经 启动 了 ， 现 在 就 可 以 打开 SQL Server 数据 库 ， 看 到 它 的 庐山 真面目 了 。 
选择 “开始 ”|“ 程 序 ”|Microsoft SQL Server 2008|SQL Server Management Studio 选项 ， 出 
现 图 1.26 所 示 界 面 。 

如 图 1.26 所 示 就 是 SQL Server 的 企业 管理 器 登录 界面 。 在 此 界面 中 , 可 以 选择 身份 验 
证 的 方式 ， 默 认 是 Windows 身份 验证 方式 不 需要 输入 密码 。 也 可 以 选择 SQL Server 身份 
验证 方式 并 输入 在 安装 时 为 用 户 设置 的 密码 。 选 择 好 登录 方式 后 ， 单 击 “ 连 接 ” 按 钮 ， 即 
可 登录 到 SQL Server 的 企业 管理 器 ， 如 图 1.27 所 示 。 


EE 
2 Koji Server2008 


服务 器 基 型 TD) [下 所 本 引 全 | 
服务 器 名 聊 (8); Wmasarwssasevizod 可 
[IT 


图 1.26 SQL Server 数据 库 登 录 界 面 图 1.27 企业 管理 器 主 界面 
至 此 ， 就 通过 上 面 讲述 的 两 步 走 的 方式 登录 到 企业 管理 器 的 主 界面 上 了 。 


1.5 了 解 SQL Server 的 工作 平台 
在 登录 到 SQL Server 2008 企业 管理 器 后 , 下面 就 带 读者 来 认识 企业 管理 器 中 每 部 分 都 


是 做 什么 的 。 
在 图 1.27 所 示 界面 中 ， 从 上 至 下 依次 是 菜单 栏 、 工 具 栏 和 工作 区 。 工 作 区 左 侧 是 对 象 


委 站 这 二 
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资源 管理 器 ， 右 侧 则 是 操作 时 显示 界面 的 位 置 。 菜 单 栏 和 工具 栏 不 用 多 说 ， 读 者 也 清楚 是 
干什么 的 ， 就 是 用 来 选择 相应 操作 的 器 。 

对 象 资源 管理 器 对 读者 来 说 ， 应 该 是 一 个 比较 陌生 的 工具 了 。 它 实际 上 就 是 用 来 管理 
数据 库 中 的 对 象 ， 包 括 数据 库 、 安 全 性 、 服 务 器 对 象 、 复 制 、 管 理 以 及 SQL Server 代理 。 
在 本 书 中 的 大 部 分 内 容 都 是 操作 对 象 资源 管理 器 中 的 数据 库 文件 夹 下 的 内 容 。 此 外 ， 还 需 
要 特别 说 明 的 是 书写 SQL 语句 的 位 置 ， 那 就 需要 单 击 “ 新 建 查询 ”按钮 ， 出 现 图 1.28 所 
示 界 面 。 


Microsoft SQL Serve 
ETTEETETOICTETT RE 
2 mes | Dy TT TT 
sd "THY a v7 D2 世 


5 QueryLsql nstrator (52))| 


图 1.28 ”打开 查询 窗口 


在 此 ， 需 要 注意 两 个 地 方 ， 一 个 地 方 是 写 着 master 的 列表 框 ， 它 代表 的 是 当前 正在 使 
用 的 数据 库 名 称 是 master， 可 以 通过 单 击 下 拉 列 表 框 ， 在 列表 中 选择 当前 要 使 用 的 数据 库 
名 称 。 另 一 个 地 方 就 是 在 对 象 资源 管理 器 右边 的 空白 区 域 , 它 是 用 来 书写 SQL 语句 的 地 方 ， 
在 后 面 的 章节 中 会 经 常 使 用 它 。 


1.6 在 SQL Server 中 已 经 存在 的 数据 库 


在 安装 好 SQL Server 数据 库 后 , 系统 会 为 其 自 带 4 个 系统 数据 库 ， 比 如 在 图 1.28 中 所 
使 用 的 master 数据 库 。 查 看 系统 数据 库 ， 通 过 图 1.27 所 示 界 面 ， 依 
次 展开 “数据 库 ”|“ 系 统 数 据 库 ” 节 点 ， 即 可 查看 到 系统 数据 库 了 ， 
如 图 1.29 所 示 。 

从 图 1.29 所 示 的 界面 中 ， 可 以 看 出 系统 数据 库 包括 master、 
model、msdb 以 及 tempdb。 下 面 就 分 别 讲解 这 4 个 数据 库 具体 的 
作用 。 


1. master 数据 库 


该 数据 库 主要 记录 SQL Server 的 系统 级 信息 ， 包 括 元 数据 、 端 点 、 链 接 服务 器 和 系统 
配置 设置 。 此 外 ,还 记录 了 所 有 其 他 数据 库 的 存在 、 数 据 库 文件 的 位 置 以 及 SQL Server 的 
初始 化 信息 等 内 容 。 因 此 ，master 数据 库 被 删除 后 ， 数 据 库 系统 就 无 法 启动 了 。 


2. model 数据 库 
该 数据 库 主 要 用 作 在 SQL Server 实例 上 创建 的 所 有 数据 库 的 模板 。 因此 ，model 数 


图 1.29 系统 数据 库 
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据 库 也 是 不 能 够 删除 的 。 
3. msdb 数据 库 


该 数据 库 主要 是 在 计划 警报 和 作业 , SQL Server Management Studio 、Service Broker 和 
数据 库 邮 件 等 功能 使 用 。 

4. tempdb 数据 库 

该 数据 库 是 一 个 全 局 资源 ， 可 用 于 保存 临时 对 象 、 数 据 库 引 擎 创建 的 内 部 对 象 以 及 日 
志 信息 等 内 容 。 通 过 该 数据 库 可 以 减少 日 志 信 息 所 占用 的 资源 ， 提 高 数据 库 访 问 的 速度 。 
此 外 ， 需 要 注意 的 是 tempdb 数据 库 是 不 能 够 备份 和 还 原 的 。 


1.7 本 章 小 结 


本 章 主要 讲解 了 与 数据 库 相 关 的 一 些 概念 以 及 安装 数据 库 的 流程 、 如 何 打开 SQL 
Server 的 企业 管理 器 等 内 容 。 在 安装 数据 库 时 ， 主 要 以 在 Windows XP 环境 下 安装 为 例 进 
行 讲解 ， 但 是 ， 读 者 也 可 以 试 着 在 其 他 Windows 环境 下 安装 。 企 业 管 理 器 作为 SQL Server 
数据 库 的 重要 组 成 部 分 ， 主 要 讲解 了 它 的 界面 各 部 分 的 内 容 。 


1.8 本 章 
一 、 填 空 题 
1. 常见 的 数据 库 有 
2. 数据 库 管 理 系 统 的 缩写 是 8 
3. SQL Server 的 登录 方式 有 种 。 
二 、 选 择 题 
1. 下 面 对 SQL Server 2008 描述 正确 的 是 6 


A. 该 数据 库 可 以 安装 到 任意 的 操作 系统 中 
B. 该 数据 库 可 以 免费 安装 到 任意 的 操作 系统 中 
C. 该 数据 库 可 以 免费 安装 到 Windows 操作 系统 中 
D. 该 数据 库 仅 能 安装 到 Windows 系列 的 操作 系统 中 
2. 下 面 对 SQL Server 2008 登录 描述 正确 的 是 
A. 该 数据 库 不 用 启动 任何 服务 就 可 以 直接 登录 
B. 该 数据 库 只 能 使 用 用 户 名 和 密码 的 方式 登录 
C. 该 数据 库 只 能 使 用 Windows 用 户 登录 方式 登录 
D. 以 上 都 不 对 
3. 下 面 对 系统 数据 库 描述 正确 的 是 g 
A. 系统 数据 库 是 指 在 安装 SQL Server 后 自 带 的 数据 库 ， 可 以 将 其 删除 
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B. 系统 数据 库 是 指 在 安装 SQL Server 后 自 带 的 数据 库 ， 不 能 将 其 删除 
C. 系统 数据 库 可 以 不 安装 
D. 以 上 都 不 对 
、 问 答题 
. 简 述 数据 库 系 统 与 数据 库 管 理 系统 的 关系 。 
简 述 在 安装 SQL Server 时 需要 注意 的 问题 。 
SQL Server 的 4 个 系统 数据 库 都 有 什么 作用 ? 
、 操 作 题 
1. 在 网 上 下 载 与 操作 系统 匹配 的 SQL Server 数据 库 ， 并 安装 。 


2. 分 别 使 用 不 同 的 登录 方式 登录 SQL Server 数据 库 。 
3. 使 用 企业 管理 器 查看 系统 数据 库 。 


LO iD 一 


旦 


。20 。 
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仓库 是 用 来 存放 各 种 物品 的 地 方 ， 管 理 仓库 的 人 员 被 称 为 仓库 管理 员 。 那 么 ， 存 储 数 
据 的 仓库 就 是 我 们 要 学 习 的 数据 库 了 。 管 理 数据 库 的 人 员 ， 就 是 数据 库 管 理 员 了 。 存 储 物 
品 的 仓库 可 以 分 为 存放 图 书 的 仓库 、 存 放 食用 油 的 仓库 、 存 放 医 疗 器 械 的 仓库 等 等 。 数 据 
库 也 是 一 样 的 ， 存 储 不 同 用 途 的 数据 ， 就 要 创建 不 同名 称 的 数据 库 。 

本 章 的 主要 知识 点 如 下 : 

口 如 何 创 建 数 据 库 

口 如 何 修改 数据 库 

口 如 何 删 除数 据 库 


2.1 创建 数据 库 


除了 SQL Server 中 的 系统 数据 库 之 外 ， 在 使 用 其 他 数据 库 时 ， 第 一 步 都 是 要 创建 数据 
库 。 通 常情 况 下 ， 都 是 确定 了 要 在 数据 库存 放 哪 些 数 据 之 后 再 创建 数据 库 。 这 样 数据 库 的 
名 字 就 可 以 根据 存放 的 数据 用 途 进行 命名 ， 使 其 更 有 实际 意义 了 。 在 SQL Server 中 创建 数 
据 库 既 可 以 通过 SQL 创建 , 也 可 以 在 企业 管理 器 中 直接 创建 。 创建 数据 库 的 具体 方法 和 注 
意 事项 将 在 下 面 的 内 容 中 一 一 道 来 。 


2.1.1 创建 数据 库 的 语法 


在 创建 数据 库 之 前 ， 读 者 有 没有 想 过 数据 库 的 结构 是 什么 样 的 呢 ? 现在 告诉 你 答案 
吧 ，SQL Server 中 的 数据 库 道 常 由 数据 文件 和 事务 日 志 组 成 ， 一 个 数据 库 可 以 由 一 到 多 个 
数据 文件 和 事务 日 志 组 成 。 数 据 文件 就 是 存储 数据 的 地 方 ， 而 事务 日 志 是 用 来 记录 存储 数 
据 的 时 间 和 操作 的 ， 通 常 可 以 根据 事务 日 志 来 恢复 数据 库 中 的 数据 。 因 此 ， 不 能 够 随便 将 
事务 日 志文 件 删 掉 。 在 本 书 中 研究 的 数据 库 通 常 是 由 一 个 数据 文件 和 一 个 事务 日 志文 件 组 
成 的 。 数 据 文件 的 扩展 名 是 .mdf， 而 事务 日 志文 件 的 扩展 名 是 .ldf， 以 后 读者 看 到 扩展 名 就 
应 该 知道 是 数据 库 中 哪 种 类 型 的 文件 了 。 创 建 数据 库 的 一 般 语 法 如 下 : 

CREATE DATABASE database name 

[ ON 
[ PRIMARY ] [ Ne | | 


BH <ftilegroup> ni 
LOG ON { <filespec> [ ,...n ] }] 


[ 
其 中 : 
口 database_ name: 数据 库 名 称 。 数 据 库 名 称 不 能 以 数字 开头 ， 一 般 是 以 英文 单词 或 
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缩写 或 汉语 拼音 来 命名 的 。 可 以 用 中 文 命名 ， 但 是 不 推荐 读者 使 用 。 

口 ON: 显 式 定义 用 来 存储 数据 库 数 据 部 分 的 数据 文件 。 如 果 省 略 了 ON 语句 ， 系 统 
也 会 可 以 默认 创建 一 个 数据 库 。 数 据 库 的 数据 文件 和 日 志文 件 都 与 数据 库 的 名 称 
一 样 ， 只 是 扩展 名 不 同 而 已 。 这 些 文件 会 存储 到 数据 库 安装 的 默认 路 径 中 。 

口 PRIMARY: 主 数据 文件 。 所 谓 主 数据 文件 就 是 在 创建 数据 库 时 指定 的 第 一 个 数据 
文件 。 一 个 数据 库 中 只 能 有 一 个 主 数据 文件 。 其 他 的 数据 文件 被 称 为 次 要 数据 


交 件 。 
口 LOG ON: 日 志文 件 。 如 果 没 有 对 数据 指定 日 志文 件 ， 系统 也 会 为 其 自动 创建 一 个 
日 志文 件 。 


当然 ， 创 建 数 据 库 的 语法 并 非 上 面 给 出 的 这 么 简单 的 儿 旬 话 ， 还 有 儿 个 子 句 也 可 以 运 
用 在 数据 库 的 创建 语句 中 。 这 里 ， 只 是 为 了 让 读者 先 认识 一 下 创建 数据 库 的 语句 ， 在 下 一 
小 节 的 内 容 中 还 会 涉及 一 些 这 里 未 提 及 的 语句 。 要 注意 看 呈 ! 


2.1.2 用 最 简单 的 语句 创建 数据 库 


所 谓 最 简单 的 语句 ， 就 是 将 上 一 小 节 中 给 出 的 创建 数据 库 的 语法 只 保留 必要 的 关键 
字 ， 其 他 的 由 系统 自动 来 创建 。 也 就 是 只 保留 CREATE DATABASE database_name 这 样 一 
句 话 就 可 以 了 。 

【示例 1】 创建 一 个 名 为 chapter2 的 数据 库 。 

用 最 简单 的 语法 创建 ， 语 句 如 下 : 


CREATE DATABASE chapter2; 


执行 上 面 的 语句 ， 效 果 如 图 2.1 所 示 。 四 
通过 上 面 的 语句 创建 好 的 数据 库 ， 应 该 是 在 安装 数据 库 EEC 


软件 时 的 默认 位 置 ， 也 就 是 在 Microsoft SQL Server 2meamw 让 | 古本 名 | 入 | 吕 回 马 沪 


MSSQL10.MSSQLSERVER2008\MSSQLWDATA 文件 夹 下 面 。 | 本 证 ear。 让 
在 这 个 文件 夹 下 面 会 有 两 个 与 chapter2 相关 的 数据 库 文件 ， shine 

请 读者 自己 找 一 找 。 这 两 个 数据 库 文件 一 个 是 数据 文件 | 地 $m 
chapter2.mdf， 一 个 是 日 志文 件 chapter2_log.LDF。 TraseT Wninistrete oster |00:00.00 [0 1 


9 行 1 列 28 Ch26 肥 


2.1.3 ”为 数据 库 指定 一 个 位 置 图 2.1 创建 chapter2 数据 库 


为 了 方便 数据 库 管 理 员 管理 ， 同 时 也 便于 查找 数据 库 ， 通 常会 在 创建 数据 库 时 为 数据 
库 指定 一 个 位 置 。 在 创建 数据 库 时 为 数据 指定 的 位 置 ， 读 者 一 定 要 记 清 楚 了 ， 否 则 就 很 难 
找到 了 。 为 数据 库 指定 位 置 不 仅 可 以 指定 数据 文件 的 位 置 也 可 以 指定 日 志文 件 的 位 置 。 此 
外 ， 还 可 以 更 细 化 到 数据 文件 的 大 小 、 自 动 增长 量 等 信息 。 

【示例 2】 创建 一 个 名 为 chapter2_1 的 数据 库 。 并 将 其 数据 文件 保存 在 d:\database 文 
件 夹 下 。 

在 创建 数据 库 之 前 ， 要 先 确保 DD 盘 下 的 database 文件 夹 存在 ， 如 果 不 存 在 请 读者 自己 
创建 一 个 ， 否 则 就 会 出 现 错误 了 ! 创建 该 数据 库 的 语句 如 下 : 


CREATE DATABASE chapter2 1 
ON PRIMARY 
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name= chapter2 1 data, 
filename='d:\database\ chapter2 1 data.mdf', 
size=3MB, 
maxsize=20M, 
filegrowth=10% 

) 

LOG ON 

《 
name= chapter2 1 log, 
filename='d:\database\ chapter2 1 log.ldf', 
size=512KB, 
maxsize=10MB, 
filegrowth=10% 

) 


执行 上 面 的 语句 ， 就 可 以 完成 数据 库 chapter2_1 的 创建 了 。 


ET 视图 WW 查询 @) 
了 新建 查询 局 信也 四 让 扇 回 3 县 5 


项 目 @) 调式 @) 工具 中 窗口 四 社区 局 帮助 oD 


BY 8 | ester | 执行 C) b》 及 叹 澡 


-数据 文件 的 逻辑 名 称 
-数据 文件 的 存放 位 置 
-数据 文件 大 小 
-数据 文件 的 最 大 值 
-- 数 据 文件 的 增长 量 


=-- 日 志文 件 的 逻辑 名 称 
=-- 日 志文 件 的 存放 位 置 
-= 日 志文 件 大 小 

-- 日 志文 件 的 最 大 值 
=-- 日 志文 件 的 增长 量 


效果 如 图 2.2 所 示 。 


=|I 寺 


Ee Ie 


SQLdueryl. s... ator (52))* 


-x 


CREATE DATABASE ch 
ON PRINARY 


2_1_data, 


Vdaeabane\ chapter?_1_data mde' , 


database\chapter2_1_log. 1df' 


WIDW-9527\RSSQLSERVER2008 ( . ， WIDW-9527\Adninistrato. 


- 款 据 文件 的 梁 畔 各 移 
-六 文人 拘 人 

-- 数 据 文件 大 
-六 作息 大 入 
一 数据 文件 的 增长 量 


master |00:00;00 |0 行 


行 17 列 2 


Ch2 


Js 


图 2.2 创建 数据 库 chapter2_1 


如 果 在 创建 时 没有 创建 database 文件 夹 ， 就 会 出 现 如 图 2.3 所 示 的 错误 提示 。 读 者 可 


以 试 试 是 否 会 出 现 这 个 错误 呢 ? 


| 

谭 息 s133， 级 别 16， 状 态 1， 第 
对 文件 村 

洽 1802， 级 别 1 Ea 

CREATE DATABASE hg. 是 列 隐 亲 ww 广 件 名 . 请 查看 相关 错误 。 


工行 
terz_1_daca.mdt” 的 目录 查找 失败 ， 出 现 操作 系统 错误 z 系 统 找 不 到 指定 的 文件 。)。 i 


一 一 一 一 一 有 


作 查询 已 完成 ， 但 有 错误 。 | YIDY-9527\ISSQLSERVER2008 ( 


| YITDY-9527\Adaninistrate .. 


master |00:00:00 |0 行 


匹配 : ( 行 15 列 52 


Ch 52 Ins /4 


图 2.3 没有 database 文件 夹 时 创建 数据 库 的 错误 提示 


另外 ， 文 件 的 单位 不 仅 可 以 为 KB 或 MB， 
长 量 部 分 也 可 以 不 使 用 百分比 的 形式 ， 


:在 创建 数据 库 时 ， 文 件 的 大 小 一 定 要 大 于 512K， 否 则 就 无 法 成 功 创建 数据 库 。 
还 可 以 是 GB、TB 等 单位 。 
可 使 用 KB 或 MB 作为 其 单位 。 


在 文件 增 


如 果 不 能 


预测 数据 库 的 最 大 容量 ， 可 以 将 maxsize 的 值 设置 成 Unlimited (无 限制 ) 。 


2 


.4 


在 示例 2 中 已 经 学 习 了 如 何在 创建 数据 库 时 指定 数据 文件 的 存储 位 置 了 ， 那 么 ， 如 何 
在 创建 数据 库 时 指定 多 个 数据 文件 呢 ? 实际 上 ， 与 示例 2 的 方法 大 同 小 异 ， 只 是 在 创建 第 
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创建 由 多 文件 组 成 的 数据 库 


一 个 数据 文件 的 后 面 ， 再 加 上 一 个 或 多 个 数据 文件 。 同 理 ， 
个 数据 库 中 会 有 无 数 个 数据 文件 吗 ? 当然 不 是 了 ， 最 多 就 是 32767 个 文件 。 


【示例 3】 创建 一 个 名 为 chapter2_2 的 数据 库 。 该 数据 库 由 两 个 数据 文件 和 两 个 日 志 


文件 构成 。 


为 了 便于 读者 查找 数据 库 ， 仍 然 将 该 数据 库 创建 在 d:\database 文件 夹 下 。 具 体 的 创建 


语句 如 下 : 


CREATE DATABASE chapter2 2 
ON PRIMARY 


在 di\database 文件 夹 下 应 该 有 4 个 与 chapter2 2 数据 库 相 关 的 数据 文件 ， 如 图 2.5 所 示 。 
全 注意 : 数据 库 中 的 文件 都 是 不 能 重 名 的 。 实 际 上 ， 一 个 数据 库 只 有 一 个 主 数据 文件 ， 后 


| 


) 


name= chapter2 21 data, 
filename='d:\database\ chapter2 21 data.mdf', 
size=3MB, 

maxsize=20MB, 

filegrowth=10% 


name= chapter2 22 data, 
filename='d:\database\ chapter2 22 data.ndf', 
size=3MB, 

maxsize=20MB, 

filegrowth=10% 


LOG ON 


( 


。24。 


name= chapter2 21 log, 
filename='d:\database\ chapter2 21 log.1df', 
size=512KB, 

maxsize=10MB, 

filegrowth=10% 


name= chapter2 22 log, 
filename='d:\database\ chapter2 22 log.1ldf', 
size=512KB, 

maxsize=10MB, 

filegrowth=10% 


日 志文 件 也 是 一 样 。 那 么 ， 一 


-- 数 据 文件 的 逻辑 名 称 
=-- 数 据 文件 的 存放 位 置 
-- 数 据 文件 大 小 

-- 数 据 文件 的 最 大 值 
-- 数 据 文件 的 增长 量 


=-- 数 据 文件 的 多 辑 名 称 
-- 数 据 文件 的 存放 位 置 
=-- 数 据 文件 大 小 

=-- 数 据 文件 的 最 大 值 

-- 数 据 文件 的 增长 量 


=-- 日 志文 件 的 逻辑 名 称 
=-- 日 志文 件 的 存放 位 置 
-= 日志 文件 大 小 

=-- 日 志文 件 的 最 大 值 
=-- 日 志文 件 的 增长 量 


=-- 日 志文 件 的 逻辑 名 称 
-- 日 志文 件 的 存放 位 置 
-= 日 志文 件 大 小 

-- 日 志文 件 的 最 大 值 
=-- 日 志文 件 的 增长 量 


执行 上 面 的 语句 ， 就 可 以 创建 chapter2 2 数据库 了 。 效 果 如 图 2.4 所 示 。 
执行 上 面 的 语句 后 ,查看 一 下 d:\database 文件 夹 ,是 否 能 找到 刚才 创建 的 数据 文件 呢 ? 


缓 名 是 .mdf。 其 他 数据 文件 就 称 为 次 要 数据 文件 ， 其 文件 后 组 是 .ndf. 
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文件 四 ”篇 辑 上 ”视图 人 )， 查 淘 册 ”项 目 E) 调式 ) 工具 中) 窗口 四 社区 C) 帮助 o 

卫 新 旭光 有 | 记 | 孔 孙 怠 | 应 | 写 吕 本 | 骂 二 

于 起 | "ster -| ?执行 g_P mv 弹 林 加 | 开 码 | 多 轩 丰 ) 三 全 | 这 李 | 人 各 昌 
SQLQueryl. s... ator (52))*| vx 


CREATE DATABASE chapterz 2 
ON PRINARY 


ter2_21_data, -- 数 据 文件 的 逻辑 名 称 
+:\database\ Chapter2_21_data.maf 玉生 六 作 的 刘 放 位 置 


人 
一 数据 文件 的 增长 量 


图 chapter2_21_data maf 
国 =hapter2_21_log 1df 
园 查 已 成 功 执行. MTDY-952TWISSQLSERYEH2006 (| TIY-952TWAdainistrate | waster |00:00:00 |0 行 团 -hapter2_22_data ndf 
P:( 行 3 到 1 ol ms 围 chapter? 22_1o8 1df 


图 2.4 创建 chapter2 2 数据 库 图 2.5 ”chapter2 2 数据 库 的 文件 


2.1.5 通过 文件 组 也 能 创建 数据 库 


文件 组 这 个 语句 没有 在 创建 数据 库 的 语法 中 提 及 ， 但 是 在 创建 数据 库 时 却 用 到 了 这 个 
概念 。 文 件 组 从 字面 上 理解 ， 就 是 在 文件 组 中 存放 多 个 文件 。 在 每 一 个 数据 库 中 都 可 以 存 
在 多 个 文件 组 。 其 中 , 一 定 会 有 一 个 主 文件 组 ,其 他 的 文件 组 就 是 用 户 自 定义 的 文件 组 了 。 
那么 ， 用 户 自 定 义 文件 组 有 什么 用 呢 ? 通过 指定 自 定义 文件 组 ， 就 可 以 指定 在 文件 组 中 存 
放 的 数据 文件 了 。 如 果 没 有 自 定义 文件 组 ， 数 据 文件 会 自动 地 划分 到 主 文 件 组 中 。 换 句 话 
说 ， 文 件 组 为 数据 库 管 理 员 管理 数据 文件 提供 了 方便 。 在 创建 数据 库 时 使 用 文件 组 的 语法 
与 创建 多 个 数据 文件 的 数据 库 类 似 ， 请 读者 根据 示例 4 的 练习 先 自己 体会 一 下 吧 。 

【示例 4】 创建 一 个 名 为 chapter2 _3 的 数据 库 。 并 在 该 数据 库 中 创建 一 个 自 定义 的 文 
件 组 。 

将 该 数据 库 还 存放 在 d:database 文件 夹 下 ， 有 具体 创建 的 语句 如 下 : 


CREATE DATABASE chapter2 3 


ON PRIMARY 

( 
name= chapter2 3 data, -数据 文件 的 逻辑 名 称 
filename='d:\database\ chapter2 3 data.mdf', -数据 文件 的 存放 位 置 
size=3MB, =- 数据 文 件 大 小 
maxsize=20MB, -数据 文件 的 最 大 值 
filegrowth=10% 一- 数据 文件 的 增长 量 

), 

FILEGROUP chapter2 group -文件 组 的 名 字 


UG 
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name= chapter2 31 data, -- 数 据 文件 的 逻辑 名 称 
filename='d:\database\ chapter2 31 data.ndf', -数据 文件 的 存放 位 置 
size=3MB, 一 -数据 文件 大 小 
maxsize=20MB, -数据 文件 的 最 大 值 
filegrowth=10% =-- 数 据 文件 的 增长 量 

) 

LOG ON 

( 
name= chapter2 3 log, -日 志文 件 的 逻辑 名 称 
filename='d:\database\ chapter2 3 log.ldf', -日 志文 件 的 存放 位 置 
size=512KB, =-- 日 志文 件 大 小 
maxsize=10MB, =-- 日 志文 件 的 最 大 值 
filegrowth=10% =-- 日 志文 件 的 增长 量 

由 

执行 上 面 的 语句 ， 就 可 以 创建 数据 库 chapter2_3。 效 果 如 图 2.6 所 示 。 


micresoft SQL Server Wanagenent Studio 
文件 四 编辑 @) 视图 W)， 查询 @) 项 上 中 调 工 @) 工具 由 窗口 社区 人 
也 .关于 查询 中 | 局 | 六 六 囊 | 局 | 苞 回 汗 | 加 


帮助 0 


Wy oa | nester -| ?135 》 a v 吕 铝 国 |” 鸡 | 如 图 如 | 三 全 | 亨 人 | 怨 且 
SQLQueryl. s... ator (52))#| Se 
CREATE DATABASE chapter2 3 本 
ON PRINARY 
name= chapter2_3_data -数据 文件 的 逻辑 名 称 
filename='d:\database\ chapter2_ 3_data.mdf' 数据 文件 的 存放 位 置 
sire=3HB, -数据 文件 大 小 
mnaxsize=20MB, -数据 文件 的 最 大 值 
filegrowch=10 数据 文件 的 增长 量 
FILEGROUP chapter?_group -文件 组 的 名 字 
name= chapter2_31_data, -数据 文件 的 逻辑 名 称 
£1lenewe=!'d:\datebese\ chapter3_ 31 daca,ndar - 教 据 文件 的 存放 位 置 
size=3NB, -数据 文件 大 小 
maxsize=20NB, -数据 文件 的 最 大 值 
filegrowth=10 -数据 文件 的 增长 量 
LOG oN 
name= chapter2_3_log, 
filename™'d:\database\ chapter2_3_log.1df', 


sire=512K! 
maxsize=10NB, 
filegrowth=10: 


已 消息 | 

命令 已 成 功 完成 Es| 
园 查询 已 成 功 执行 | TY-s527WISSQLSERYEE2006 ( .| WIDW-9527\Adninistrato.. ,| master |00:00:00 |0 行 
匹配 : ( 各 攻 列 1 Ch1i Js 有 4 


图 2.6 创建 数据 库 chapter2 3 
以 发 现 使 用 文件 组 时 只 需要 使 用 FILEGROUP filegroup_name 语句 


从 本 例 中 ， 读 者 可 
来 定义 ， 其 括号 中 的 定义 与 定义 数据 文件 是 相同 的 。 另 外 ， 
据 文 件 的 数量 一 致 ， 可 


全 说 明 : 如 果 想 将 在 数据 库 中 创建 的 对 象 保存 在 自 定义 的 


以 将 多 个 数据 文件 的 日 志 信息 存放 到 同一 


kr ten ed 
日 志文 件 中 。 


文件 组 中 ， 可 以 在 定义 文件 组 时 


加 上 DEFAULT 关键 字 , 这 样 就 会 将 该 文件 组 设置 成 默认 文件 组 ,设置 方法 如 下 : 


FILEGROUP chapter2 group DEFAULT 


。26 。 
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2.1.6 看 看 究竟 创建 了 哪些 数据 库 


通过 前 面 的 儿 个 示例 已 经 创建 了 不 少数 据 库 了 ， 那 么 ， 数 据 库 管理 员 都 是 通过 查看 创 
建 数据 库 的 文件 夹 才 了 解 到 存在 哪些 数据 库 的 吗 ?” 如果 数据 库 管 理 员 忘记 了 创建 数据 库 的 
文件 来， 数据 库 就 找 不 到 了 吗 ? 读者 在 初次 使 用 数据 库 时 也 会 常常 想到 这 些 问 题 吧 ， 下 面 
就 告诉 你 儿 个 简单 的 方法 ， 来 查看 已 经 创建 的 数据 库 以 及 相关 的 数据 库 信息 。 

【示例 5】 使 用 存储 过 程 sp_helpdb 查看 全 部 数据 库 。 

存储 过 程 sp_helpdb 能 够 查看 到 所 有 的 数据 库 ， 包 括 系统 自 带 的 数据 库 和 用 户 自 定义 
的 数据 库 。 查 询 效果 如 图 2.7 所 示 。 


文件 四 ”编辑 ”视图 WD 查询 @) 项 目 @) 调 坛 @) 工具 四) 窗口 中 社区 C) 帮助 如 
:也 新建 查询 | 由 | 也 也 本 | 锯 | 写 加 本 虽 天 


|? 执行 O ， 和 nv 允 可 国政 杷 | 拉 圈 如 | 三 | 素 这 | 名 


dbid 
WIDW.9527Wdministato 8 10202012 Status=ONLINE. Updateabily=READ_WRITE. Usel 
WIDW-3527\Administalot 9 10202012 Siatus=ONLINE, Updateabily=READ_WRITE, Usel 
WIDW.9527\Administiator 10 10202012 Status=ONLINE. Updateabily=READ_WAITE. Usel 
WIDW-3527\Administiator 11 10202012 Status=ONLINE, Updateabily=READ_WRITE, Use 

7 

1 

和 

4 

5 

6 

2 


WIDW .9527Wdministator 09242012 Status=ONUINE.Updateabiily=READ_WRITE, Usel 
3 


04 82003 ”Status=ONUNE.Updateabiiy=READ_WRITE. Usel 
04 82003 Status=ONUINE,Updateabily=READ_WRITE. Usel 
07 92008 Status=ONUNE.Updateabily=READ_WRITE. Usel 


a 
3 


[9 | ReponServersMss. 350MB WIDW.9527\Administalor 
| 于 | ReponServersMss, 306MB WIDW.9527 WAdministator 
tempdb B50MB 33 


0918202 StatusrONLINE, Updatesbily=READ_WRITE. Usel 
0918202 StatuseONLINE. UpdateabilyeREAD_WAITE. Usel 
10202012 StatuseONLINE, UpdateabillyeREAD_WRITE, Usel 


到 
团 查 询 已 成 功 执行 . | TDm-9527WSSQLSERPYEFE2008 〔 .，| MTDY-952T\Aaninistrate .，| waster | 00:00:01 | 11 行 
列 10 chio Jas 元 


图 2.7 使 用 sp_helpdb 查看 数据 库 


从 图 2.7 的 查询 结果 中 ， 读 者 就 可 以 找到 刚才 在 示例 1 一 示例 4 中 创建 的 数据 库 
chapter2、chapter2_1、chapter2 2 和 chapter2 3。 

【示例 6】 使 用 存储 过 程 sp_helpdb 查看 数据 库 chapter2 的 文件 。 

使 用 存储 过 程 sp_helpdb 查看 某 一 个 数据 库 中 的 数据 文件 ， 只 需要 在 存储 过 程 
sp_helpdb 后 面 加 上 数据 库 的 名 字 即 可 。 查 看 的 语句 如 下 : 


sp_helpdb chapter2; 


执行 上 面 的 语句 ， 效 果 如 图 2.8 所 示 。 

从 图 2.8 中 ,可 以 看 到 chapter2 数据 库 由 两 个 文件 组 成 ,一 个 是 数据 文件 chapter2, 一 
个 是 日 志文 件 chapter2_log。 

【示例 7】 查看 数据 库 chapter2 空间 的 使 用 情况 。 

查看 数据 库 的 空间 使 用 情况 ， 能 够 更 好 地 利用 数据 的 空间 。 查 看 数据 库 空间 的 使 用 情 
况 可 以 使 用 存储 过 程 sp_spaceused 来 查看 。 具 体 的 语句 如 下 : 


use chapter2 一 -指定 要 查询 的 数据 库 
exec sp_spaceused; =-- 执 行 存储 过 程 


执行 上 面 的 语句 ， 效 果 如 图 2.9 所 示 。 


Pe 


第 1 篇 走 进 SQL Server 


了 mi croseft SQL Server Nanagenent Studio -Iolxl 
文件 四 ”编辑 EE) 视图 WD 查询 @) 项 目 E) 调式 W) 工具 中 窗口 J 社区 加 帮助 中 
也 新 寻 查 询 中 | 记 也 也 可 马上 写 加 本 又 下 


Raa nester -| 执行 WW 》 av 叶 可 辐 洲 辣 | 泣 立 入 | 三 全 | 率 这 | 名 
SQLQueryl. s... ator (52))*| -x 
sp_helpdb chapter2; | 
器 结果 | 已 消息 | 
[name。 | db sze owner dbid | created | status 胜 
[oo ehapte2 | 281MB WIDW.9527\Administator 8 10202012 Status=ONUINE.Updateabiiy=READ_WRITE.UserAcc 
‘ 到 
C:\Program Files\Mictosoft SQL Serve\MSSQL10.MSS. PRIMARY 2304KB Unlimited 1024 KB 
[2 | chapter2 log 2 C:\Program Files\Mictosoft SQL Serve \MSSQL10.MSS,. NULL S76KB 2147483648KB 10% 
| YTDW-9527 ISSQLSERVER2008 ( .，| WIDW-9527\Adninistrato. .. | master |00:00:00 |3 行 


文件 @) 妨 辑 区 ) 视图 W) 查询 @) 项 目 @) 调试 @) 工具 YD) 窗口 和 ) 社区 C) 帮助 人 D 
革新 尘 走光 如 六 | 六 启 咏 局 苞 回 寺 加 忆 
| chepte2 -| ?1# 行 Ww》 有 v 允 加 国 | 下 后 | 镁 图 如 | 三 他 审 | 全 局 
‘SQLQueryl. s... ator (52))*| vx 
Buse chapcer2 

exec sp_spaceused; 


器 结果 | 2 消息 | 
[| daabase name | dalabase_sbe | unalocaledspace | 
| enapte2 |281MB 106MB 


[| eseved | daa | ndex see | unused 
[Enzteke | 4eske 632k8 98kB J 
回 查询 已 成 功 执行 . | TDY-952TNWISSQLSERYER2006 (.. | WIDW-9527\Adninistrato. .. | chapter2 | 00:00:00 |2 行 

行 1 列 13 Ch 13 Ts 用 


图 2.9 查看 数据 库 chapter2 的 空间 使 用 情况 


从 图 2.9 的 查看 结果 中 ， 可 以 得 到 数据 库 chapter2 中 数据 的 大 小 〈database_size)、 未 
分 配 的 空间 (unallocated space) 和 数据 使 用 的 容量 〈data) 等 信息 。 


2.1.7 ”使 用 企业 管理 器 创建 数据 库 


学 了 这 么 多 创建 数据 库 的 语句 ， 现 在 学 点 轻松 的 内 容 吧 。 使 用 企业 管理 器 ， 不 用 记 语 
句 照样 可 以 创建 数据 库 。 在 企业 管理 器 中 创建 数据 库 就 要 简单 得 多 了 ， 并 且 也 可 以 指定 数 
据 库 中 的 文件 数量 以 及 文件 的 大 小 等 信息 。 下 面 就 使 用 示例 8 诠释 一 下 在 企业 管理 器 中 如 
何 创 建 数 据 库 。 

【示例 8】 在 企业 管理 器 中 创建 名 为 chapter2_4 的 数据 库 。 

使 用 企业 管理 器 创建 默认 的 数据 库 只 需 分 为 如 下 两 个 步骤 即 可 。 

(1) 打开 创建 数据 库 界面 

在 企业 管理 器 中 的 对 象 资源 管理 器 里 ， 右 击 “ 数 据 库 ”选项 ， 在 弹出 的 右键 菜单 中 选 
择 “ 新 建 数 据 库 ”选项 ， 出 现 图 2.10 所 示 的 界面 。 

通过 图 2.10 所 示 的 数据 库 创建 界面 就 可 以 完成 所 有 使 用 SQL 语句 创建 数据 库 的 效果 。 

(2) 添加 数据 库 名 称 以 及 数据 库 文件 

在 图 2.10 所 示 的 界面 中 , 将 数据 库 的 名 称 chapter2_4 输入 到 “数据 库 名 称 ”文本 框 中 。 
如 果 数 据 库 文件 部 分 的 信息 不 需要 修改 直接 选择 默认 的 设置 ， 则 直接 单 击 “ 确 定 ” 按 钮 ， 
即 可 完成 数据 库 的 创建 ， 如 图 2.11 所 示 。 
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EEC alz 
连 舞 页 -和 
Ey 
村 文人 组 数据 库 名 称 加- 上] 
所 有 者 加)- [SV EE 
也 所 用 全 文 素 5J07 


服务 器 
WIDW-9527 \WSSQLSERVER2006 


连接 ， 
TD-9527WAdnini strator 


于 查看 连接 尾 性 


图 2.10 创建 数据 库 界面 


Dw 


jl 
仿生 项 E x 
字 交 | 盾 数据 这 从 旭 : FTC 
所 有 站) [E> 
民 俩 月 宇 7 本 引 朵 
数 硕 宗 文 件 由: 
自动 增长 

chapter2 4 行 数 琵 PETRART 3 | 协 量 方 1 B ,不 限制 霹 长 C \Froeran Files\Mier 
chapter2 日 去 不 适 月 1 据 量 为 10%， 不 限制 增 攻 C NProgran Files\Mier 

| 

服 竺 器 : 

rss 

连 关 | 

a 

台 ， 查 连接 必 性 

| 
mn | 
添加 4) be 
Cw | ™ 
3 


图 2.11 使 用 默认 设置 创建 数据 库 chapter2_4 
至 此 ， 数 据 库 chapter2_4 已 经 创建 完成 了 。 
在 企业 管理 器 中 ， 除 了 可 以 使 用 默认 设置 创建 数据 库 之 外 ， 也 可 以 完成 更 改 其 数据 文 


件 的 大 小 、 文 件 的 存放 位 置 以 及 添加 文件 、 使 用 文件 组 等 操作 。 下 面 就 把 这 些 操作 的 具体 
方法 向 读者 一 一 道 来 。 
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1. 在 创建 数据 库 时 ， 更 改 默认 文件 的 大 小 和 存放 位 置 


(1) 更 改 文件 的 大 小 

在 图 2.10 所 示 的 界面 中 ， 如 果 要 更 改 文件 的 初始 大 小 ,直接 在 单元 格 中 更 改 即 可 。 如 
果 要 更 改 文件 自动 增长 情况 ， 需 要 单 击 “ 自 动 增长 ”单元 格 后 的 按钮 ， 弹 出 如 图 2.12 所 示 
界面 。 

在 图 2.12 所 示 的 界面 中 ， 可 以 更 改 文件 自动 增长 的 设置 以 及 最 大 文件 大 小 的 设置 。 

(2) 更 改 文件 存放 的 位 置 

如 果 想 改变 文件 存放 的 位 置 ， 需 要 在 图 2.10 所 示 的 界面 中 ， 单 击 “ 路 径 ” 单 元 格 后 面 
的 按钮 ， 弹 出 如 图 2.13 所 示 界 面 。 


前 定位 文件 夹 -YIDY-9527\ESSQLSERYER2008 


图 2.12 更 改 文件 的 自动 增长 设置 图 2.13 更 改 文件 路 径 


在 图 2.13 所 示 的 界面 中 ， 选 择 一 个 要 存放 数据 库 的 文件 路 径 ， 单 击 “ 确 定 ” 按 钮 ， 即 
可 完成 文件 路 径 的 修改 。 


2. 在 创建 数据 库 时 ， 添 加 其 他 数据 文件 


在 创建 数据 库 时 ， 默 认 的 情况 下 一 个 数据 库 是 由 一 个 数据 文件 和 一 个 日 志文 件 组 成 
的 。 如 果 要 添加 数据 文件 或 者 日 志文 件 ， 需 要 在 图 2.10 所 示 的 界面 中 , 单 击 “ 添 加 ”按钮 ， 
弹出 如 图 2.14 所 示 的 界面 。 

在 图 2.13 所 示 的 界面 中 , 如 果 要 添加 的 文件 是 数据 文件 , 在 文件 类 型 中 选择 “ 行 数据 ”; 
如 果 要 添加 的 文件 是 日 志文 件 ， 则 在 文件 类 型 中 选择 “日 志 ”。 新 添加 的 数据 文件 也 可 以 更 
改 文件 的 大 小 以 及 文件 的 存储 路 径 。 
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图 2.14 添加 数据 文件 界面 


3. 在 创建 数据 库 时 ， 创 建文 件 组 并 使 用 


在 图 2.10 中 ,细心 的 读者 已 经 看 到 了 在 数据 库 文件 的 列表 中 ， 有 一 个 文件 组 的 列 ， 在 
列 中 有 PRIMARY 和 “不 适用 ”两 个 类 型 。 PRIMARY 代表 的 是 主 数据 文件 组 ， 是 默认 的 
数据 文件 存放 类 型 。“ 不 适用 ”是 日 志文 件 的 默认 类 型 。 那 么 ， 如何 创建 用 户 自 定义 的 文件 
组 呢 ? 可 以 选择 图 2.10 左 侧 窗 口中 的 “文件 组 ” 弹出 图 2.15 所 示 的 界面 。 


EE 
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图 2.15 文件 组 界面 
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在 图 2.14 所 示 界 面 中 ， 显 示 的 是 当前 数据 库 中 存在 的 文件 组 信息 。 目 前 只 有 一 个 
PRIMARY 文件 组 。 如 果 要 添加 新 的 文件 组 ， 单 击 “ 添 加 ”按钮 ， 并 填 入 相应 的 文件 组 信 
息 即 可 。 假 设 文件 组 名 为 chapter2 4 _group， 并 设置 成 默认 组 。 效 果 如 图 2.16 所 示 。 

创建 后 的 文件 组 如 何 使 用 呢 ? 在 图 2.10 中 的 文件 组 选项 里 就 可 以 将 数据 文件 添加 到 
新 建 的 文件 组 中 了 。 读 者 可 以 自己 试 试看 哟 ! 


ET3 
EE 


= = 
服务 器 
en 


活 接 
NII-9527 Wdninistr tor 


于 坦 看 这 控 尾 考 


就 


图 2.16 创建 文件 组 chapter2 4 _group 


2.2 ”修改 数据 库 


在 现实 生活 中 ， 大 部 分 人 性 化 的 操作 都 是 可 以 修改 的 。 比 如 : 购买 一 件 衣服 后 ， 由 于 
大 小 或 者 颜色 不 合适 ， 到 商场 中 几乎 都 是 可 以 调换 的 。 对 于 数据 库 的 操作 也 不 例外 ， 也 会 
给 数据 库 操作 者 机 会 来 修改 它 。 修 改 数据 库 的 操作 包括 修改 数据 的 名 称 、 修 改 数据 库 中 数 
据 文 件 存放 的 位 置 、 修 改 数据 文件 的 大 小 以 及 添加 数据 文件 等 操作 。 在 本 节 中 虽然 没有 给 
出 一 个 完整 的 修改 数据 库 的 语句 ， 但 是 读者 如 果 看 到 ALTER DATABASE 语句 一 定 要 知道 
这 是 对 数据 库 中 的 信息 进行 修改 的 语句 。 下 面 就 几 个 常见 的 修改 数据 库 问 题 做 以 下 讲解 。 


2.2.1 给 数据 库 改名 


当 数 据 库 创建 完成 后 ， 发 现 数据 库 的 名 字 不 符合 命名 要 求 时， 可 以 通过 如 下 两 个 方法 
来 更 改 数据 库 的 名 称 。 


1. 使 用 ALTER DATABASE 语句 更 改 


使 用 ALTER DATABASE 更 改 数据 名 的 具体 语法 如 下 : 
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ALTER DATABASE old database name 

MODIFY NAME=new_ database name; 

其 中 : 

口 old_database_name: 原来 的 数据 库 名 称 。 

口 new_database name: 更 改 后 的 数据 库 名 称 。 

【示例 9】 将 数据 库 chapter2 的 名 字 更 改 成 chapter2_new。 
原来 的 数据 库 名 是 chapter2， 更 改 后 的 名 称 是 chapter2_new。 更 改 语句 如 下 : 


ALTER DATABASE chapter2 
MODIFY NAME=chapter2 new; 


执行 上 面 的 语句 后 ， 效 果 如 图 2.17 所 示 。 
现在 数据 库 中 就 不 存在 名 为 chapter2 的 数据 库 了 ， 而 换 成 了 chapter2_new 数据 库 。 


2. 使 用 存储 过 程 sp_renamedb 更 改 
过 存储 过 程 更 改 数 据 库 名 称 与 使 用 ALTER DATABASE 语句 的 效果 是 一 样 的 ， 但 是 

i sp_renamedb 存储 过 程 会 简单 一 些 。 只 需要 记 住 这 个 存储 过 程 的 名 字 就 行 了 。 语 法 规 
则 如 下 : 

Sp_renamedb old database name,new database name; 

其 中 : 

口 old_database_name: 原来 的 数据 库 名 称 。 

口 new_ database_name: 更 改 后 的 数据 库 名 称 。 


【示例 10】 将 数据 库 chapter2_new 的 名 称 更 改 成 chapter2 。 
读者 可 以 发 现 ， 示 例 10 就 是 将 数据 库 名 字 又 改 回 了 chapter2。 更 改 的 语句 如 下 : 


sp_renamedb chapter2 new, chapter2; 


执行 上 面 的 语句 ， 效 果 如 图 2.18 所 示 。 
至 此 ， 数 据 库 的 名 字 又 改 回 chapter2 了 。 


TE 
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卫 新 建 查询 叫 | 记 | 也 也 怠 | 记 | 区 四 局 日 


Rsmi cresoft SQL Server Nanceeneaele 
Er 编辑 也 ) 视图 YW) 查询 @) 项目) 
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辽 新建 查 询 中 六 | 售 六 蔬 | 让 | 芒 回 寺 


时 让 | wster =| ?执行 Oh 筷 HR ster -| ?执行 OP 各 
SQLQueryl. s- - -ater (52))* SQlQueryl-s- - -ater (52))*| vx 
ALTER DATIBASE chapter2 ap_renamedb chapter?2_new, chapter? 习 
MODIFY NANE=chapt 了 
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消息 2 Ss 
i ‘chapeerz, 数据 库 名 称 “chaprerz' 已 设置. 习 | 
加 2 和 
C.. |¥IDW-9527\Adninistrato. .. |master | 00:00:00 |0 行 | | mw-sszrWaninistrate .. | waster |00:00:00 |0 行 


就 绪 本 8 到 1 i 就 第 行 1 列 36 ch36 天 


图 2.17 使 用 ALTER DATABASE 语句 更 改 数据 库 名 称 ”图 2.18 使 用 sp_renamedb 更 改 数据 库 名 称 


2.2.2 ”给 数据 库 换个 容量 
数据 库 的 容量 是 通过 数据 库 中 的 数据 文件 的 大 小 来 确定 的 。 更 改 数据 库 的 容量 也 就 是 
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更 改 数据 库 中 数据 文件 的 大 小 。 具 体 语法 如 下 : 


ALTER DATABASE database name 

MODIFY FILE 

( 
NAME=datafile name, 
NEWNAME=new_datafile name, 
FILENAME="'file path', 
SIZE=new size, 
MAXSIZE=new_maxsize, 
FILEGROWTH=new filegrowth 


晶 中 : 

database_name: 数据 库 名 称 。 

NAME: 数据 文件 名 。 也 就 是 要 修改 的 数据 文件 名 称 。 

NEWNAME: 更 改 后 的 数据 文件 名 。 如 果 不 需要 修改 数据 文件 的 名 称 ， 
以 省 略 。 


口 口 口 


该 语句 可 


口 SIZE: 数据 文件 的 初始 大 小 。 如 果 不 需要 修改 数据 文件 的 初始 大 小 ， 该 语 名 可 以 


省 略 。 


口 MAXSIZE: 数据 文件 的 最 大 值 。 如 果 不 需要 修改 数据 文件 的 最 大 值 ， 该 语句 可 以 


省 略 。 


口 FILEGROWTH: 文件 自动 增长 值 。 如 果 不 需 要 修改 文件 自动 增长 值 ， 该 语句 可 以 


【示例 11】 将 数据 库 chapter2 中 的 数据 文件 的 初始 大 小 改 成 30M。 
根据 题目 要 求 ， 只 更 改 数据 文件 chapter2 的 初始 大 小 即 可 。 更 改 的 语句 如 下 : 
ALTER DATABASE chapter2 
MODIFY FILE 
( 
NAME=chapter2, =-- 数 据 文件 名 


SIZE=30MB -数据 文件 的 初始 大 小 
) 


执行 上 面 的 语句 ， 效 果 如 图 2.19 所 示 。 


这 样 ，chapter2 数据 库 中 的 数据 文件 的 初始 值 就 更 改 成 。“ 剖 使 加 如 3@ 到 


调试 四) 工具 CO) 窗口 如 社区 局 帮助 人 D 


了 30MB。 通 过 sp_helpdb 可 以 验证 一 下 修改 的 效果 ， 如 2sww | 局] 有 久久 | 号 加 


时 到 | wester 


-| 口交 | 


图 2.20 所 示 。 | A 
通过 图 220 的 查询 结果 ， 可 以 看 出 chapter2 数据 库 中 的 “这 

数据 文件 的 初始 大 小 已 经 更 改 成 了 30720KB。 那 么 ， 读 者 会 

想 明明 是 修改 成 30M 怎么 会 是 30720KB 呢 ? 实际 上 ,这 个 “| 

结果 是 正确 的 ,因为 1IM=1024KB。 请 读者 自己 换算 一 下 ,30M | bs 


NANE: 


是 多 少 KB 呢 ? re 
就 绪 行列 2Ch2 性 
2.2.3 在 数据 库 中 添加 文件 图 2.19 更 改 数据 的 初始 大 小 


除了 改变 原 有 的 数据 库 设 置 之 外 ， 还 可 以 在 数据 库 中 添加 数据 文件 或 者 是 日 志文 件 、 
文件 组 。 在 向 数据 库 中 添加 文件 前 ， 先 要 通过 存储 过 程 sp_helpdb 来 查看 一 下 现 有 的 文件 
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eicrosoft SQL Server Hanageaeat Stodie -可 
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“SQLQueryl.s... ator (52))*| vx 
sp_helpdb chapterd 习 
器 结果 | 二 消息 | 
rame [db se owner J 是 | seaed [status compatibily, 
记 chapte2 3056 MB WIDW-9527\sdministator 8 10202012 Status=ONLINE., Updateabiity=READ_WRITE. Userhcc- 100 
name | fed | flenane EE Marsie Wage 
加 chapte2 | 1 C:\Program Files\Microsoft SQL Server\MSSQLIOMSS.. PRIMARY | 30720KB | Unimied 1024KB dataonly 
NY) -hape 2 __ Ci\PrgramFies\Micro:of SQL Serve\MSSQLIOMSS.. NULL S75KB 21474. 10% kgonky 到 
回 查询 忆 成行 - NID-9527\SSQLSFRVER2008 〔 .| 了 DY-952TWdninistrate ，|waster |00:00:00 |3 行 
L 行 ! 列 1 ED ED 


图 2.20 查看 修改 后 的 chapter2 数据 文件 的 初始 大 小 
人 信息。 这样， 在 向 数据 库 添加 新 文件 时 就 可 以 避免 重 名 了 。 下 面 先 学习 一 下 向 数据 库 中 添 
加 文件 的 语法 规则 吧 。 


ALTER DATABASE database name 
[ADD FILE|LOG FILE 
( 


NAME=logic file name, 
FILENAME='file path', 
SIZE=new_size, 
MAXSIZE=new maxsize, 
FILEGROWTH=new filegrowth 
)] 
ADD FILEGROUP filegroup name 
[TO FILEGROUP filegroup name] 


其 中 : 

口 database name: 数据 库 名 称 。 也 就 是 要 修改 的 数据 库 名 称 。 

口 ADD FILE: 添加 数据 文件 。 添 加 数据 文件 和 创建 日 志文 件 的 文件 结构 是 一 样 的 ， 
这 里 就 不 再 解释 了 。 

口 ADD LOG FILE: 添加 日 志文 件 。 

口 ADD FILEGROUP: 添加 文件 组 。 

口 TO FILEGROUP: 为 数据 文件 指定 文件 组 。 如 果 没 有 指定 文件 组 ， 默 认 情况 下 ， 
数据 文件 会 添加 到 PRIMARY 文件 组 中 。 

下 面 通 过 示例 12 和 示例 13 来 讲解 如 何 应 用 上 面 的 语法 来 添加 文件 或 文件 组 。 

【示例 12】 在 chapter2 数据 库 中 添加 一 个 名 为 chapter2_newadd 的 数据 文件 。 

在 添加 数据 文件 之 前 先 通过 sp_helpdb 查看 一 下 chapter2 数据 库 的 存储 位 置 以 及 数据 

文件 的 名 字 。 添 加 chapter2_newadd 数据 文件 的 语句 如 下 : 


ALTER DATABASE chapter2 

ADD FILE 

( 
NAME=chapter2 newadd, -数据 文件 名 
FILENAME="'C:\ProgramFiles\MicrosoftSQL 

Server\MSSQL10 .MSSQLSERVER2008\MSSQL\DATA\chapter2_1.ndf', -- 文 件 存储 位 置 


SIZE=3MB, -- 文 件 初始 大 小 
MAXSIZE=UNLIMITED, =-- 文 件 的 最 大 值 
FILEGROWTH=10% =-- 文 件 的 自动 增长 率 
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执行 上 面 的 语句 ， 效 果 如 图 2.21 所 示 。 
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图 2.21 向 chapter2 数据 库 中 添加 数据 文件 


添加 数据 文件 chapter2_newadd 后 ，chapter2 数据 库 当 前 的 数据 文件 组 成 如 图 2.22 
所 示 。 


er 
有 调 吉 工具 加 
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CNProgan FlesMeroso SOLSeveWSSOLIOMSS PRIMARY™ 30720KB “Unimied 1024 KB data onl 
chapet bg 2 CPromanFleAMerosokSOLSeve\WSSOLIOMSS. NULL 57KB 21474. 10% -loonb 
chapler? maewadd © 3 C:\Program lociMersoR SDL Seve\MSSOLIOMSS.. PRIMARY 3072K3 Uninied 10% coorb 
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图 2.22 chapter2 数据 库 添 加 数据 文件 后 的 效果 


【示例 13】 在 数据 库 chapter2 中 添加 一 个 名 为 chapter2_group 的 文件 组 ， 然 后 再 为 数 
据 库 添 加 一 个 名 为 chapter2_newadd1 的 数据 文件 ， 并 将 该 文件 添加 到 chapter2_group 文件 
组 中 。 

根据 题目 要 求 ， 先 创建 文件 组 ， 再 将 新 创建 的 数据 文件 添加 到 文件 组 中 。 具 体 的 语句 
如 下 : 


ALTER DATABASE chapter2 


ADD FILEGROUP chapter2 group =-- 添 加 文件 组 
ALTER DATABASE chapter2 
ADD FILE =-- 添 加 文件 
( 
NAME=chapter2 newaddl, =-- 添 加 的 数据 文件 名 


FILENAME='C:\ProgramFiles\MicrosoftSQL 
Server\MSSQL10 .MSSQLSERVER2008\MSSQL\DATA\chapter2 2.ndf', 
SIZE=3MB, 
MAXSIZE=UNLIMITED, 
FILEGROWTH=10% 
to filegroup chapter2 group -- 将 数据 文件 添加 到 chapter2_group 文件 组 中 


执行 上 面 的 语句 ， 效 果 如 图 2.23 所 示 。 
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行 13 列 和 0 EE ms 


图 2.23 添加 文件 组 并 将 数据 文件 添加 到 该 文件 组 中 


在 chapter2 数据 库 上 完成 了 添加 文件 组 和 数据 文件 后 ，chapter2 数据 库 中 的 数据 文件 
信息 如 图 2.24 所 示 。 
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图 2.24 ”chapter2 数据 库 中 文件 信息 


从 图 2.24 中 可 以 看 到 ， 在 chapter2 数据 库 的 文件 信息 中 多 了 一 个 数据 文件 chapter2_ 
newadd1， 并 且 该 数据 文件 在 chapter2_group 文件 组 中 。 

在 完成 了 示例 12 和 示例 13 的 练习 后 ， 在 chapter2 数据 库 中 添加 日 志文 件 对 于 读者 来 
说 应 该 是 小 菜 一 碟 了 。 下 面 就 请 读者 自己 试 试 如 何在 chapter2 数据 库 中 添加 日 志文 件 吧 。 


2.2.4 在 数据 库 中 清理 无 用 文件 


数据 库 中 的 文件 不 能 一 味 地 添加 ， 当 数据 库 中 的 某 些 文件 不 再 需要 时 ， 就 需要 清理 一 
下 数据 库 中 的 文件 了 。 在 上 一 小 节 中 学 习 了 如 何 添加 数据 文件 、 日 志文 件 以 及 文件 组 。 那 
么 ， 在 本 小 节 中 就 来 学 习 一 下 如 何 删除 这 些 文件 吧 。 删 除 文件 可 要 简单 得 多 哟 ! 删除 这 些 
文件 的 语法 如 下 : 


ALTER DATABASE database name 
REMOVE FILE|FILEGROUP file name|filegroup name 


其 中 : 
口 database name: 数据 库 名 称 。 
口 REMOVE FILE: 移 除 文件 。 移 除 的 文件 包括 数据 文件 和 日 志文 件 。 移 除 文件 只 需 
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要 在 该 语句 后 面 加 上 要 移 除 的 文件 名 称 即 可 。 

口 REMOVE FILEGROUP: 移 除 文件 组 。 移 除 文件 组 的 前 提 是 要 确保 文件 组 中 没有 
任何 文件 。 需 要 移 除 文件 组 时 ， 只 需要 将 文件 组 的 名 字 加 在 该 语句 之 后 即 可 。 

下 面 就 将 示例 12 和 示例 13 中 添加 的 数据 文件 和 文件 组 删除 掉 。 

【示例 14】 把 数据 文件 chapter2_ newadd 从 chapter2 数据 库 中 删除 。 

根据 题目 要 求 ， 删 除数 据 文件 的 语句 如 下 : 


ALTER DATABASE chapter2 
REMOVE FILE chapter2 newadd; -- 要 删除 的 数据 文件 名 


执行 上 面 的 语句 ， 效 果 如 图 2.25 所 示 。 


x ver Nenagenent Studio 
文件 虽 ”篇 辑 区 ) 视图) 查询 @) 项 目 E) 调式 Wm) 工具 G 窗口 中 


区 加 者 助人 9 
和 2 寻 喜光 如 | 出 也 也 瑟 上 由 |[ 蕊 思 林 又 二 
时 纪 | ee -1436@ 》 nv 弛 避 回 了 码 时 


SqlQuery2. =- -ater (53; 
站 ALTER DATABASE ch 
RENOVE FILE ch 


Sodoeryl. sator E2)) | a 
~- 要 删除 的 数据 文件 名 


: 
De | 


文件 'chapcerz_nevadd， 已 删除 。 a 
| » 


| WIDW-9527 \WSSQLSERYER2008 ( .，| WIDW-9527\Adninistrato. .. | master |00:00:00 |0 行 
[2 行 2 列 29 Ch 4 


图 2.25 删除 数据 文件 chapter2_newadd 


至 此 ， 数 据 文件 chapter2_newadd 就 从 chapter2 数据 库 中 移 除 了 。 

【示例 15】 把 文件 组 chapter2_group 从 chapter2 数据 库 中 删除 。 

根据 题目 要 求 ， 要 删除 文件 组 chapter2_group。 由 于 在 文件 组 chapter2_group 中 存在 数 
据 文件 chapter2_newadd1， 因 此 要 先 将 其 删除 ， 再 删除 文件 组 。 语 句 如 下 : 


ALTER DATABASE chapter2 


REMOVE chapter2 newadd1 -删除 数据 文件 
ALTER DATABASE chapter2 
REMOVE FILEGROUP chapter2 group =-- 删 除 文件 组 
执行 上 面 的 语句 ， 效 果 如 图 2.26 所 示 。 
ss IE] 
WD 查询 @) 项 目 @) 调 坛 WD) 工具 WD 窗口 W 社区 EC) 帮助 0 
全 亲本 | 让 | 也 也 到) 记 | 碟 回忆 | 骂 呈 
a eester -| A5W 由 w 33 吾 回 | 3 网 | 办 辆 四) 三 全 | 音字 | 已 昌 
L9n .到 EE vx 
-一 除数 据 文 伯 3 
-日 除 文件 组 
下 下 » 
已 消息 | 
文件 'chapterz_nevaddl， 已 删除 。 划 
文件 组 ,chapeezz_sroup， 已 删除 - = 
| ; 
固 豆 河 已 成 功 执行 。 | Yip-9527 \nssQLSERVER2008 (.. | ITDY-952T\Adninistrate .，| waster |00:00:00 |0 行 
Ee 看 二 列 3 扣 Ch25 Ins 4 


图 2.26 删除 文件 组 chapter2_group 
至 此 ， 文 件 组 chapter2_group 就 从 数据 库 chapter2 中 移 除了 。 
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2.2.5 ”使 用 企业 管理 器 修改 数据 库 


在 前 面 的 学 习 中 已 经 知道 了 如 何在 企业 管理 器 中 创建 数据 库 ， 那 么 ， 在 企业 管理 器 中 
修改 数据 就 更 容易 掌握 了 。 在 企业 管理 器 中 修改 数据 库 ， 也 涵盖 了 使 用 语句 修改 数据 库 的 
全 部 操作 。 但 是 ， 修 改 数据 库 的 名 字 与 其 他 的 修改 操作 略 有 不 同 。 下 面 就 分 两 个 方面 来 讲 
解 在 企业 管理 器 中 修改 数据 库 的 方法 。 


1. 修改 数据 库 的 名 字 


修改 数据 库 的 名 字 很 简单 ， 直 接 右 击 要 修改 名 字 的 数据 库 ， 在 弹出 的 右键 菜单 中 选择 
“ 重 命名 ”选项 ， 然 后 在 数据 库 名 字 处 输入 新 的 数据 库 名 字 ， 按 回 车 键 确认 即 可 。 读 者 可 以 
自行 尝试 将 chapter2 改 成 chapter2_new。 


2. 修改 数据 库 中 的 文件 以 及 文件 组 


修改 数据 库 中 的 文件 及 文件 组 与 创建 数据 库 时 的 方法 类 似 ， 这 里 以 修改 chapter2 数据 
库 为 例 进行 讲解 。 

(1) 打开 数据 库 属性 界面 

在 企业 管理 器 的 对 象 资源 管理 器 中 ， 右 击 chapter2 数据 库 ， 在 弹出 的 右键 菜单 中 选择 
“属性 ”选项 。 弹 出 如 图 2.27 所 示 的 数据 库 属性 界面 。 


区 
字 事 务 E 才 传 渤 


服务 器 
TY-952TVISSOLSERYER2008 


连接 
NIN-9527 \Adninistrator 


再” 坦 看 连接 性 


就 绪 


名 称 
效 扎 库 的 名 各 


wl 
图 2.27 chapter2 数据 库 属性 界面 


(2) 修改 文件 并 确定 
在 图 2.27 所 示 界 面 中 ， 选 择 “ 文 件 ” 选 项 ， 出 现 chapter2 数据 库 中 的 文件 信息 ， 如 
图 2.28 所 示 。 
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2.28 ”修改 文件 界面 


在 图 2.28 所 示 的 界面 中 ， 读 者 会 发 现 与 之 前 创建 数据 库 的 界面 是 类 似 的 ， 可 以 添加 或 
删除 数据 文件 和 日 志文 件 。 有 具体 的 操作 这 里 就 不 再 袭 述 了 。 只 是 在 进行 任何 操作 后 ， 要 单 
击 “ 确 定 ”按钮 ， 保 存 所 做 的 修改 。 

(3) 修改 文件 组 并 确定 

在 图 2.27 所 示 界 面 中 ， 选 择 “ 文 件 组 ”选项 ， 出 现 chapter2 数据 库 中 的 文件 组 信息 ， 
如 图 2.29 所 示 。 


2.29 ”修改 文件 组 界面 
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在 图 2.29 所 示 界 面 中 ， 可 以 添加 或 删除 chapter2 中 的 文件 组 。 但 是 ， 在 企业 管理 器 中 
删除 文件 组 时 也 要 注意 ， 只 有 文件 组 中 没有 数据 文件 了 才能 够 删除 。 在 对 文件 组 进行 操作 
后 ， 也 要 记得 单 击 “ 确 定 ” 按 钮 ， 保 存 对 chapter2 数据 库 中 文件 组 所 做 的 修改 。 


2.3 ”删除 数据 库 


当 数 据 库 中 的 所 有 数据 文件 都 不 再 需要 时 ， 整 个 数据 库 就 没有 用 了 。 这 时 ， 就 需要 删 
除 整 个 数据 库 而 不 再 是 某 个 数据 文件 了 。 但 是 ， 读 者 要 清楚 ， 删 除 后 的 数据 库 就 不 能 够 再 
恢复 了 ， 因 此 ， 在 删除 前 一 定 要 对 数据 库 中 的 信息 进行 备份 。 删 除数 据 库 可 以 说 是 数据 库 
操作 中 最 简单 的 一 个 环节 了 ， 本 节 也 是 最 轻松 的 了 。 


2.3.1 小 试 使 用 语句 删除 数据 库 


删除 数据 库 的 语句 非常 简单 ， 使 用 DROP DATABASE 语句 就 可 以 删除 。 具 体 的 语法 
格式 如 下 : 


DROP DATABASE database name; 


其 中 ，database_name 就 是 要 删除 的 数据 库 名 称 。 如 果 不 清楚 要 删除 的 数据 库 名 ， 可 以 
通过 sp_helpdb 来 查看 数据 库 。 

【示例 16】 删除 名 为 chapter2 的 数据 库 。 

根据 题目 的 要 求 ， 已 经 知道 了 要 删除 的 数据 库 名 ， 其 语句 如 下 : 


DROP DATABASE chapter2; 


执行 上 面 的 语句 ， 效 果 如 图 2.30 所 示 。 
至 此 ， 数 据 库 chapter2 已 经 被 移 除 了 。 人 TD 
调试 四 工具 CD) 窗口 中 社区 避 ) 和 助 0 
2.3.2 ”使 用 企业 管理 器 删除 数据 库 Da DLL a 
SalQuery2.s.. .ator (53))#| ZX 
使 用 语句 删除 数据 库 都 是 一 名 话 的 事 ， 那 么 ， 在 企业 管 | 
理 器 中 删除 数据 库 就 更 便捷 了 。 只 需要 在 对 象 资源 管理 器 中 | 号 对) 
右 击 要 删除 的 数据 库 ， 在 弹出 的 右键 菜单 中 选择 “删除 ” 先 mw 
项 即 可 将 数据 库 删除 了 。 无 论 用 哪 种 方法 删除 数据 库 ， 读 者 是 A 
一 定 要 记得 先 备 份 数 据 库 再 进行 删除 。 读 者 可 以 自己 在 企业 图 2.30 删除 chapter2 数据 库 
管理 器 中 试 着 删除 一 个 数据 库 ， 体 验 一 下 企业 管理 器 给 你 带 
来 的 方便 吧 。 


2.4 本 章 小 结 


在 本 章 中 主要 讲解 了 通过 SQL 语句 和 企业 管理 器 如 何 创 建 、 修 改 以 及 删除 数据 库 。 在 
创建 数据 库 时 要 注意 文件 大 小 的 设置 以 及 文件 的 存储 位 置 ; 在 修改 数据 库 时 要 注意 不 能 将 
数据 文件 的 大 小 设置 成 比 修改 前 的 还 小 ， 以 及 不 能 够 直接 清除 含有 文件 的 文件 组 ; 删除 数 
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据 库 是 最 简单 的 操作 ， 但 是 也 要 在 删除 之 前 备份 数据 库 ， 以 避免 不 必要 的 损失 。 


2.5 本 章 习 题 


一 、 填 空 题 
1. 数据 库 中 主 数据 文件 的 扩展 名 是 5 
2. 数据 库 通常 由 和 文件 组 成 。 
3. 删除 数据 库 使 用 的 语句 是 
二 、 选 择 题 
1. 创建 数据 库 test， 语 句 是 
A. create data test B. create database test C. 以 上 都 不 对 
2. 下 面 对 数 据 库 的 描述 正确 的 是 


A. 一 个 数据 库 只 能 有 一 个 数据 文件 和 一 个 日 志文 件 
B. 一 个 数据 库 只 能 有 一 个 数据 文件 和 多 个 日 志文 件 
C. 一 个 数据 库 可 以 有 多 个 数据 文件 和 多 个 日 志文 件 
D. 以 上 都 不 对 
3. 下 面 对 修 改 数据 库 的 描述 正确 的 是 
A. 不 能 给 数据 库 改名 
B. 在 数据 库 创 建 完成 后 ， 不 能 随意 更 改 数据 库 的 大 小 
C. 可 以 使 用 系统 存储 过 程 sp_renamedb 更 改 数据 库 的 名 称 
D. 以 上 都 不 对 


、 问 答题 


1. 如 何 给 数据 库 更 改 容量 大 小 ? 
2. 如 何 给 数据 库 添加 文件 组 ? 
.如 何 删除 数据 库 中 的 文件 ? 


四 、 操 作 题 


1. 创建 一 个 名 为 testl 的 数据 库 。 
2. 为 testl 数据 库 使 用 SQL 语句 添加 数据 文件 。 
3. 分 别 使 用 两 种 方法 将 数据 库 testl 的 名 字 改 成 test2。 


LD 
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在 第 2 章 中 已 经 讲解 了 数据 库 的 一 些 基本 操作 ， 那 么 ， 数 据 库 中 的 数据 是 如 何 存放 的 
呢 ? 数据 库 就 相当 于 是 一 个 文件 夹 ， 在 一 个 文件 夹 中 可 以 存放 多 个 文件 。 数 据 库 中 的 文件 
被 称 为 数据 表 ， 也 就 是 用 来 存储 数据 的 容器 。 一 个 数据 库 由 若干 张 数据 表 组 成 ， 每 张 数据 
表 的 名 字 都 是 唯一 的 ， 就 像 一 个 文件 夹 中 的 文件 名 都 是 唯一 的 一 样 。 

本 章 的 主要 知识 点 如 下 

口 数据 表 中 的 数据 类 型 
口 如 何 创 建 数 据 表 
口 如 何 修改 数据 表 
口 如 何 删 除数 据 表 


3.1 认识 表 中 能 存放 什么 样 的 数据 


读者 可 以 思考 一 下 ， 当 我 们 在 网 站 上 注册 一 个 用 户 信 息 的 时 候 ， 都 会 输入 哪些 数据 
呢 ? 通常 会 有 用 户 名 、 密 码 、 邮 箱 、 年 龄 和 联系 方式 等 等 。 只 要 是 注册 时 填 入 的 数据 ， 最 
终 都 将 提交 到 数据 库 中 存放 。 想 想 这 些 数据 都 包含 什么 呢 ? 输入 注册 信息 的 时 候 会 有 汉字 、 
数字 、 字 母 以 及 特殊 符号 等 。 既 然 这 些 数据 都 够 存 到 数据 库 中 ， 也 就 是 说 数据 表 中 应 该 能 
够 存放 这 些 类 型 的 数据 。 在 本 节 中 将 详细 讲解 SQL Server 数据 表 中 使 用 的 数据 类 型 。 


3.1.1 ” 整 型 和 浮 点 型 


整 型 和 浮 点 型 实际 上 都 属于 数值 类 型 ， 也 就 是 用 来 存放 数字 的 一 种 类 型 。 这 个 类 型 在 
日 常生 活 中 用 得 是 比较 多 的 ， 读 者 想 一 想 在 什么 情况 下 需要 整数 和 小 数 呢 ?” 当 存放 年 龄 时 
需要 整数 ， 当 存放 金额 时 需要 小 数 ， 当 存放 商品 数量 时 需要 整数 等 等 。 这 样 看 来 整数 和 浮 
点 数 很 重要 上 , 那 就 让 我 们 看 看 在 SQL Server 数据 库 中 究竟 整数 和 浮 点 数 用 什么 数据 类 型 
名 表示 的 吧 ! 首先 ， 来 学 习 一 下 表 3-1 所 示 的 整数 类 型 。 


表 3-1 整数 类 型 


数据 类 型 取 值 范围 说 了 明 

bit | 存储 0 或 ! 表示 位 整数 ， 除 了 0 和 1 之 外 ， 也 可 以 取 值 NULL 
tinyint | 0-28-1 表示 小 整数 ， 占 1 个 字 节 

smallint 25 一 25-1 表示 短 整 数 ， 占 2 个 字 节 


_231~_23L_1 


表示 一 般 整数 ， 占 用 4 个 字 节 
表示 大 整数 ， 占 用 8 个 字 节 


int 


_26~26_1 


bigint 
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从 表 3-1 可 以 看 出 ， 整 数 类 型 主要 包括 bit、tinyint、smallint、int 和 bigint， 它 们 的 取 
值 范围 是 从 小 到 大 的 。 在 实际 的 应 用 中 ， 要 根据 存储 数据 的 大 小 选择 数据 类 型 ， 这 样 能 够 
节省 数据 库 的 存储 空间 。 这 就 像 你 在 超市 结账 时 , 根据 选择 物品 的 多 少 来 购买 购物 袋 一 样 。 
如 果 购 买 的 东西 多 ， 就 选择 大 号 的 ， 如 果 购 买 的 东西 少 ， 就 选择 小 号 的 。 

接 下 来 让 我 们 再 认识 一 下 表 3-2 所 示 的 浮 点 型 吧 。 


表 3-2 浮 点 型 


说 明 
表示 -1038+1~1038-1 范围 中 的 任意 小 数 , numeric(m,n) 中 的 m 
代表 有 效 位 数 ，n 代表 小 数 要 保留 的 小 数位 数 。 例 如 : 
numeric(7,2) 表 示 长 度 为 7 的 数 ， 并 保留 2 位 小 数 

与 numeric(m,n) 的 用 法 相同 

占用 4 个 字 节 


占用 8 个 字 节 


数据 类 型 取 值 范围 


numeric(m,n) | -1038+1~1038-1 


decimal(m.n) | -103+1~1038-1 


-3.40E+38~3.40E+38 


从 表 3-2 可 以 看 出 ， 如 果 要 精确 表示 小 数 可 以 使 用 numeric(m,n) 或 者 decimal(m,n); 如 
果 不 需 要 精确 并 且 表 示 更 多 的 小 数位 数 ， 可 以 使 用 real 或 者 float。 总 之 是 要 根据 数据 的 大 
小 和 精度 选择 合适 的 浮 点 型 。 
3.1.2 字符 串 类 型 

字符 串 类 型 是 数据 表 中 存储 数据 最 常用 的 数据 类 型 。 那 么 ， 什 么 样 的 数据 可 以 用 字符 
串 类 型 来 表示 呢 ? 实际 上 任何 数据 都 可 以 说 成 是 字符 串 类 型 ， 汉 字 、 字 母 、 数 字 、 一 些 特 
殊 字 符 甚 至 是 日 期 形式 都 可 以 用 字符 串 类 型 来 存储 。 用 来 表示 字符 串 的 数据 类 型 是 按照 存 
储 字符 串 的 长 度 划分 的 。 有 具体 分 类 如 表 3-3 所 示 。 
表 3-3 字符 串 类 型 


数据 类 型 取 值 范围 说 有 明 
二 | 用 来 表示 同 定 长度 的 字符 惠 ， 如 果 存 放 的 数据 没有 达到 定义 的 长 

charl(n) 1 一 8000 个 字符 | 度 ， 系 统 会 自动 用 空格 填充 到 该 长 度 

Vaehaa | 18000 个 字符 ”| 用 于 表示 变 长 的 数据 。1 个 字符 点 1 个 字 节 ， 不 用 空格 填充 长 度 


用 于 表示 变 长 的 数据 。 该 数据 类 型 表示 的 长 度 是 输入 数据 的 实际 
长 度 加 上 2 字 节 

用 于 表示 变 长 的 数据 。1 个 字符 占 1 个 字 节 ， 最 大 可 以 存储 2GB 
的 数据 

用 于 表示 固定 长 度 的 双 字 节 数 据 。1 个 字符 占 2 个 字 节 。 与 char 
类 型 一 样 ， 如 果 存 放 的 数据 没有 达到 定义 时 长 度 ， 系 统 会 自动 
空格 填充 到 该 长 度 

用 于 表示 变 长 的 数据 。 与 varchar(n) 的 区 别 就 是 1 个 字符 需要 占 
2 个 字 节 来 表示 

用 于 表示 变 长 的 数据 。 该 数据 类 型 表示 的 长 度 是 输入 数据 的 实际 
长 度 的 2 倍加 上 2 字 节 

用 于 表示 变 长 的 数据 。1 个 字符 占 2 个 字 节 ， 最 大 可 以 存储 2GB 
的 数据 


varchar(max) | 1~~2>-1 个 字符 


text 1~231-1 个 字符 


nchar(n) 1 一 4000 个 字符 


nvarchar(n) 1 一 4000 个 字符 


nvarchar(max) | 1 一 23-1 个 字符 


ntext 1~23!_1 个 字符 
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数据 类 型 取信 范围 说 明 
3 用 于 表示 固定 长 度 的 二 进 制 数据 。 如 果 输 入 数据 的 长 度 没有 达到 
binary) | 1~8000 个 字符 | 定义 的 长 度 ， 用 0X00 填充 
| 用 于 定义 一 个 变 长 的 数据 。 存 储 的 是 三 进 制 数据 ， 答 入 的 数据 实 
aminay) 18000 个 字符 。 | 际 长 度 小 于 定义 的 长 度 也 不 需要 十 充值 
1 1 个 空竹 | 用 于 定义 一 个 变 长 的 数据 。image 类 型 不 用 指定 长 度 ， 可 以 存 依 
image 2 a 二 进 制 文 件数 据 
通过 学 习 表 3-3 中 的 数据 类 型 ， 读 者 不 难 发 现 ， 实 际 上 字符 串 类 型 可 以 大 致 上 分 为 三 
类 ， 一 类 是 1 个 字符 占用 1 个 字 节 的 字符 串 类 型 (char、varchar 和 text)， 一 类 是 1 个 字符 


占用 2 个 字 节 的 字符 串 类 型 (nchar、nvarchar 和 ntext)， 一 类 是 存放 二 进 制 数据 的 字符 串 
类 型 (binary、varbinary 和 image )。 在 每 一 类 中 字符 串 类 型 又 分 为 存放 固定 长 度 和 可 变 长 
度 的 类 型 , 在 实际 的 应 用 中 推荐 读者 使 用 可 变 长 度 的 类 型 , 这 样 可 以 节省 数据 的 存储 空间 。 


名 说明: 在 字符 串 类 型 中 的 varchar(max) 和 nvarchar(max) 类 型 是 在 SQL Server 2005 版 本 
上 开始 使 用 的 。 


3.1.3 日 期 时 间 类 型 
虽然 日 期 时 间 可 以 用 字符 串 类 型 表示 , 但 是 在 SQL Server 中 还 是 准备 了 一 套数 据 类 型 
来 专门 用 于 表示 日 期 时 间 。 通 过 日 期 时 间 类 型 可 以 将 日 期 时 间 表 示 得 更 加 准确 ， 在 SQL 
Server 中 表示 日 期 时 间 的 数据 类 型 主要 有 datetime 和 smallldatetime 两 种 。 有 具体 的 表示 方法 
如 表 3-4 所 示 。 
表 3-4 日 期 时 间 型 


数据 类 型 取 值 范 说 ”了 明 
datetime 1753 年 1 月 1 日 ~9999 年 12 月 31 日 占用 8 个 字 节 ， 精 确 到 3.33 毫秒 
smalldatetime ”| 1900 年 1 月 1 日 一 2079 年 6 月 6 日 占用 4 个 字 节 ， 精 确 到 分 钟 


虽然 有 了 存储 日 期 时 间 的 数据 类 型 ， 但 还 要 清楚 的 是 日 期 时 间 的 存储 格式 。 通 常 日 期 
的 输入 格式 有 3 种 : 英文 + 数字 的 格式 、 数 字 + 分 隔 符 的 格式 以 及 数字 格式 。 下 面 就 分 别 用 
这 3 种 形式 来 表示 2012 年 5 月 1 日 。 


May 1 2012 =-- 英 文 + 数字 格式 
2012-5-1 -- 数 字 + 分 隔 符 格式 
2 0 和 255 -- 数 字 + 分 隔 符 格式 
20127579 -- 数 字 + 分 隔 符 格 式 
20120501 -数字 格式 
120501 -数字 格式 


看 了 上 面 的 例子 ， 你 对 日 期 类 
数字 + 分 


型 的 表示 有 所 了 解 了 吧 。 其 实 ， 在 这 3 种 表示 方法 中 ， 


隔 符 的 格式 是 最 常用 的 ， 也 是 最 灵活 的 。 除 了 上 面 的 3 种 数字 + 分 隔 符 的 表示 形式 


外 ， 还 可 以 按照 月 日 年 、 日 月 年 的 顺序 来 表示 日 期 类 型 的 数据 。 例 如 : 5-1-2012、1-5-2012 


等 形式 。 
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除了 日 期 有 固定 的 存储 格式 外 ， 时 间 部 分 的 数据 也 有 固定 的 存储 格式 。 通 常 时间 类 型 
的 数据 存储 格式 都 是 按照 “小 时 : 分 钟 : 秒 .毫秒 ”来 存储 的 。 例 如 : 表示 上 午 的 9 点 10 
分 20 秒 ， 就 可 以 用 9:10:20 来 表示 。 时 间 的 表示 可 以 分 为 24 小 时 和 12 小 时 两 种 格式 ， 如 
果 用 的 是 12 小 时 的 格式 ， 用 am 表示 上 午 ， 用 pm 表示 下 午 。 例 如 : 表示 晚上 的 10 点 40 
分 10 秒 ， 就 可 以 用 10:40:10 pm 表示 。 

在 存储 日 期 时 间 数 据 时 通常 将 日 期 和 时 间 一 起 存储 ， 这 时 就 需要 在 日 期 格式 后 面 加 上 
一 个 空格 然后 加 上 时 间 格 式 来 表示 。 例 如 : 表示 2012 年 5 月 25 日 下 午 5 点 25 分 10 秒 ， 
就 可 以 写成 2012-5-25 5:25:10pm。 存 储 日 期 时 间 类 型 时 ， 读 者 要 注意 的 就 是 格式 问题 ， 另 
外 ， 还 要 提醒 读者 在 一 个 数据 表 中 存储 的 日 期 时 间 格 式 要 统一 ， 否 则 在 查询 数据 时 就 会 给 
你 造成 一 些 不 必要 的 麻烦 的 哦 ! 


3.1.4 其 他 数据 类 型 


除了 上 面 讲 的 3 类 比较 常用 的 数据 类 型 外 ， 还 有 一 些 不 太 常用 的 数据 类 型 。 比 如 : 
timestamp 类 型 、xml 类 型 和 cursor 类 型 等 。timestamp 类 型 是 时 间 惟 类型， 在 更 新 数据 时 ， 
系统 会 自动 更 新 时 间 戳 类 型 的 数据 ， 它 也 可 以 用 于 表示 数据 的 唯一 性 。 另 外 ， 在 一 张 数据 
表 中 只 能 有 一 个 时 间 惟 类 型 ，xml 类 型 可 以 存储 之 前 学 过 的 其 他 类 型 的 数据 ， 也 可 以 存储 
XML 文件 格式 的 数据 ， 它 的 存储 空间 最 大 是 2GB; cursor 类 型 是 用 于 存储 变量 或 者 是 存储 
过 程 输出 的 结果 ， 它 通常 都 用 于 存储 查询 结果 ， 在 存储 过 程 中 应 用 较 多 。 

除了 系统 自 带 的 数据 类 型 外 ， 如 果 用 户 觉 得 这 些 数 据 类 型 满足 不 了 需求 时 也 可 以 自 定 
义 数据 类 型 。 自 定义 数据 类 型 很 简单 ， 具 体 的 语句 如 下 : 


CREATE TYPE type_name 

FROM datatype; 

其 中 : 

口 type_name: 自 定义 的 数据 类 型 名 称 。 名 称 不 能 以 数字 开头 。 

口 datatype: 数据 类 型 。 定 义 自 定义 数据 类 型 表示 的 数据 类 型 ， 除 了 指定 数据 类 型 外 ， 

还 可 以 指定 该 类 型 是 否 为 空 值 。 

【示例 1】 定义 一 个 数据 类 型 ， 用 来 表示 字符 串 ， 长 度 是 20 并 且 不 能 为 空 。 

根据 题目 要 求 , 仍然 需要 定义 一 个 字符 串 类 型 , 可 以 选择 的 系统 的 字符 串 类 型 有 很 多 ， 
char、varchar、nchar 和 nvarchar 都 是 可 以 的 。 这 里 ,选择 一 个 可 变 长 度 的 字符 串 类 型 varchar。 
具体 的 语句 如 下 : 

CREATE TYPE usertypel 

FROM varchar (20) not null; 

通过 上 面 的 语句 就 可 以 为 数据 库 添加 一 个 数据 类 型 usertypel, 在 使 用 该 类 型 时 直接 用 
usertypel 就 可 以 了 。 

如 果 不 需 要 自 定义 的 数据 类 型 了 ， 也 可 以 通过 DROP TYPE 语句 将 其 删除 。 如 果 要 删 
除 在 示例 1 中 定义 的 数据 类 型 usertypel， 删 除 的 语句 如 下 : 


DROP TYPE usertypel; 


对 于 自 定义 数据 类 型 的 应 用 ， 还 将 在 下 面 的 小 节 中 详细 讲解 。 请 读者 在 下 一 节 中 认真 
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体会 自 定义 数据 类 型 的 优势 吧 ! 
3.2 ”创建 数据 表 


数据 表 在 数据 库 中 的 地 位 ， 就 好 像 是 人 的 器 官 一 样 重 要 。 如 果 人 没有 器 官 ， 那 么 人 就 
是 一 个 空 架 子 毫 无 意义 。 如 果 数 据 库 中 一 张 数据 表 都 不 在 在， 那么 数据 库 也 就 没有 了 存在 
的 意义 。 既 然 数据 表 如 此 的 重要 ， 就 让 我 们 先 学 习 一 下 数据 表 是 如 何 创建 的 吧 。 

3.2.1 创建 数据 表 的 语句 

创建 数据 表 的 语法 是 非常 复杂 的 ， 语 句 也 非常 多 。 但 是 不 用 怕 ， 咱 们 由 浅 入 深 慢 慢 来 
学 。 在 本 小 节 中 先 学 习 一 下 创建 数据 表 的 基本 语法 格式 ， 如 下 : 

CREATE TABLE table name 

( 


column namel datatype, 
column name2 datatype, 


其 中 : 
口 table_ name: 表 名 。 在 一 个 数据 库 中 数据 表 的 名 字 不 能 重复 ， 且 数据 表 不 能 用 数字 
来 命名 。 通 常 要 将 表 名 声明 成 有 实际 意义 的 名 字 。 
口 column_namel: 字段 名 。 表 中 的 字段 名 也 是 不 能 重复 的 。 
口 datatype: 数据 类 型 。 它 可 以 是 系统 的 数据 类 型 ， 也 可 以 是 用 户 自 定义 的 数据 类 型 。 
3.2.2 ”试用 CREATE 语句 创建 简单 数据 表 
有 了 在 上 一 小 节 中 讲解 的 创建 数据 表 的 语法 ， 就 可 以 创建 数据 表 了 。 但 是 ， 这 个 数据 
表 只 是 最 简单 的 一 种 形式 ， 只 有 字段 名 和 数据 类 型 ， 没 有 其 他 的 内 容 。 不 管 是 多 么 简单 的 
一 张 表 ， 都 要 先 弄 清楚 表 中 的 字段 名 和 数据 类 型 。 假 设 要 完成 一 张 用 户 信息 表 的 创建 ， 表 
的 字段 名 和 数据 类 型 用 发 表 3-5 所 示 。 
表 3-5 用 户 信息 表 (userinfo) 


编 | 字段 名 | 数据 类 型 说 阴 


和 2 一 和 一 一 varchar(20) 用 户 名 

3 password varchar(10) 密码 

4 email varchar(20) 邮箱 

5 QQ varchar(15) QQ 号 码 
6 tel varchar(15) 电话 号 码 


从 表 3-5 可 以 看 出 ， 除 了 编号 外 都 设置 成 了 字符 串 类 型 。 但 是 ， 字 符 串 类 型 的 长 度 设 
置 略 有 不 同 。 编 号 用 整数 来 表示 ， 可 以 给 其 设置 成 自动 增长 的 ， 可 以 避免 用 户 编号 重复 。 
那么 ， 读 者 会 问 了 ， 为 什么 用 户 编号 不 能 够 重复 呢 ? 其 实 ， 这 就 是 为 了 避免 出 现 多 条 重复 


号 二 这 二 
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的 记录 ， 如 果 重 复 的 话 就 很 难 判断 是 哪个 用 户 了 。 这 就 好 像 是 每 个 人 都 共用 同一 个 卡号 的 
银行 卡 ， 那 么 如 何 知 道 给 谁 发 工资 了 呢 ? 谁 花 钱 了 呢 ? 当然 ， 也 可 以 将 其 他 字段 设置 成 不 
重复 的 ， 使 用 第 4 章 中 介绍 的 唯一 约束 就 可 以 很 容易 地 设置 了 。 下 面 就 用 示例 2 来 演示 如 
何 创建 用 户 信息 表 。 
【示例 2】 根据 表 3-5 的 字段 信息 创建 用 户 信息 表 (userinfo )。 
根据 题目 要 求 ， 创 建 用 户 信息 表 的 语句 如 下 所 示 。 这 里 在 chapter3 数据 库 中 来 创建 数 
据 表 。 如 果 没 有 chapter3 数据 库 ， 请 读者 自行 创建 一 个 名 为 chapter3 的 数据 库 。 本 章 的 所 
有 数据 表 都 将 创建 在 该 数据 库 中 。 
USE chapter3 =-- 打 开 chapter3 数据 库 
CREATE TABLE userinfo 
, id inty 
name varchar(20), 
password varchar (10), 


email Varchar (20) ， 
[ole) varchar (15), 
tel varchar (15) 


有 


执行 上 面 的 语句 ， 就 可 以 在 chapter3 数据 库 中 创建 userinfo 数据 表 了 。 执 行 效果 如 
图 3.1 所 示 。 


ver 有 anagesent Studio =|Dlxl 
六 rT ET 查询 @) 项 目 @) 调 工 ) 工具 0) 窗口 思 
社区 ) 帮助 中 
也 新建 查 淘 0 让 也 也 台所 | 七 加 与 | 台电 
| ?执行 C) 》 本 v 吧 生 回 


-打开 chapter3 数 据 库 -| 


DD CREATE TABLE userinf 


id int, 
name varchar (20), 
th varchar (10) , 


varchar (20), 
oo varchar 115) , 
tel varchar (15 | 


J2T7\MSSQLSERVER2008 ( WI-9527 Adninistrato. .| chapter3 | 00:00:00 0 行 
就 绪 行 1 列 26 Ch 26 


图 3.1 创建 表 userinfo 


3.2.3 创建 带 自动 增长 字段 的 数据 表 


所 谓 自动 增长 字段 就 是 让 字段 按照 某 一 个 规律 增加 ， 这 样 就 可 以 做 到 该 列 的 值 是 唯一 
的 。 在 SQL Server 数据 库 中 , 设置 带 自动 增长 字段 的 前 提 是 该 字段 是 一 个 整数 类 型 的 数据 。 
另外 在 设置 自动 增长 字段 时 ， 还 需要 指定 最 小 值 以 及 每 次 增加 多 少 个 参数 。 具 体 的 设置 方 
式 如 下 : 


IDENTITY (minvalue,increment) 


其 中 : 
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口 minvalue: 最 小 值 , 也 可 以 说 是 该 列 第 一 个 要 使 用 的 值 。 默认 情况 下 是 从 1 开始 的 。 

口 increment: 每 次 增加 值 。 默 认 情 况 下 也 是 每 次 加 1。 

如 果 要 采用 默认 的 从 1 开始 每 次 增加 1 的 自动 增长 方式 , 直接 使 用 IDENTITY 关键 字 
设置 即 可 ， 不 需要 再 添加 参数 了 。 有 了 自动 增长 字段 的 设置 方式 ， 那 么 该 语句 应 该 写 在 什 
么 位 置 呢 ? 请 读者 先 在 示例 3 中 自己 找 一 找 。 

【示例 3】 根据 表 3-5 的 字段 信息 创建 用 户 信息 表 (userinfol1)， 并 将 该 表 中 的 编号 列 
(id) 设置 成 自动 增长 列 。 

根据 题目 要 求 ， 具 体 的 创建 表 语 句 如 下 所 示 。 仍 然 将 该 表 创 建 在 chapter3 数据 库 中 。 
由 于 该 数据 库 中 已 经 存在 了 userinfo 的 数据 表 ， 因 此 ， 将 该 表 的 名 字 定 义 成 userinfol 。 


USE chapter3 -- 打 开 chapter3 数据 库 
CREATE TABLE userinfol 
( 
id int IDENTITY(1,2), -设置 自动 增长 字段 
name varchar(20), 
password varchar (10), 
email varchar (20) ， 
[ele] varchar (15), 
tel varchar (15) 
Dh 


执行 上 面 的 语句 ， 就 可 以 在 chapter3 中 创建 表 userinfol 了 。 执 行 效果 如 图 3.2 所 示 。 


Ricrosoft SQL Server Nanagenent Studio =ID|x| 
文件 四 ”编辑 @) 视图 中 查询 @) 项 目 E) 调式) 工具 CD 窗口 中) 
社区 @C) 帮助 中 


辽 审 建 查询 m | 六 | 也 也 可 | 庙 | 蕊 加 本 | 进 瑟 
时 | chapter3 "| 执行 》 由 v 33 下 国 


SQLQueryl. s... ator (53))* et 
日 usE chapter3 -- 打 开 chapter3 数 据 库 


DCREATE TABLE userinfol 


id int IDENTITY(1,2 -- 设 置 自动 增长 字段 
varchar (20 
rd varchar (10), 

varchar (20), 

varchar (15), 


| chapter3 |00;00:00 |0 行 
列 17 EE 


fe7 NSSQLSERVER2008 〔 .，| YIDY-9527\Adninistrato. 
就 绪 行 1 


图 3.2 创建 表 userinfol 


通过 上 面 的 例子 , 相信 读者 已 经 知道 了 IDENTITY 这 个 语句 放 在 什么 位 置 了 吧 。 没 错 ， 
就 是 放 在 需要 设置 成 自动 增长 列 数据 类 型 后 面 。 
3.2.4 创建 带 自 定义 数据 类 型 的 数据 表 

读者 可 以 思考 一 下 ， 如 果 要 在 表 3-5 所 示 的 字段 信息 中 使 用 自 定 义 数据 类 型 ， 应 该 将 
哪些 字段 设置 成 自 定 义 数据 类 型 呢 ? 在 表 3-5 中 有 两 个 字段 使 用 了 varchar(15)， 有 两 个 字 
段 使 用 了 varchar(20)， 可 以 分 别 将 varchar(15) 的 定义 成 一 个 数据 类 型 ， 将 varchar(20) 的 定 
义 成 一 个 数据 类 型 。 这 样 ， 不 仅 在 这 个 表 中 ， 在 整个 数据 库 里 如 果 再 需要 这 两 种 数据 类 型 
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时 也 可 以 直接 使 用 自 定义 的 数据 类 型 。 实 际 上 ， 经 常会 将 一 个 表 或 一 个 数据 库 中 经 常 出 现 
的 数据 类 型 定义 成 自 定义 数据 类 型 。 

【示例 4】 根据 表 3-5 创建 用 户 信息 表 (userinfo2) 并 使 用 用 户 自 定义 类 型 usertypel 。 
在 创建 用 户 信息 表 之 前 ， 先 创建 一 个 自 定义 数据 类 型 usertype1， 类 型 是 varchar(15)。 

根据 题目 的 要 求 ， 先 创建 自 定义 数据 类 型 usertype1， 语 句 如 下 : 

USE chapter3 


CREATE TYPE usertypel 
FROM varchar (15); 


不 要 忘记 了 , 将 该 数据 类 型 也 创建 到 数据 库 chapter3 中 。 执行 上 面 的 语句 , 在 chapter3 
中 就 创建 了 一 个 名 为 usertypel 的 数据 类 型 。 
在 创建 用 户 信息 表 (userinfo2) 时， 使 用 usertypel 数据 类 型 ， 具 体 的 语句 如 下 : 


USE chapter3 =-- 打 开 chapter3 数据 库 
CREATE TABLE userinfo2 
( 
id int, 
name varchar(20), 
password varchar (10), 
email varchar (20) ， 
QQ usertypel, 
tel usertypel 
0 


执行 上 面 的 语句 ， 就 可 以 在 数据 库 chapter3 中 创建 表 userinfo2 了 。 执 行 效果 如 图 3.3 
所 示 。 


=I9| x| 
文件 @) 编辑 如 视图 WD 查询 Q@) 项 目 人 E) 调式 W) 工具 GD) 窗口 WD) 社区 C) 
帮助 0 
驴 新 奸 查 淘 D | 记 孔 也 下 | 应 | 蕊 呈 呈 | 过 天 
Pe -| 执行 Wy》 v 33 本 回 | 于 三 | 妈 图 
SQLQueryl ater (53))#| -x 
日 0SE cha) 


向 CREATE TT ertypel -创建 自 定义 数据 类 型 
| FRON varcha 5 


USE chapter3 
CREATE TABLE userin 


-- 打 开 chapter3 数 据 库 


ee 二 
| » 


园 查 | ITm-eszrWissqLsFRYEFE2008 ( .| WIDW-9S27\Adninistrato, .| chspter3 |00:00:00 |0 行 
就 绪 行 1 


列 19 ch ll ms /| 


图 3.3 创建 表 userinfo2 


3.2.5 在 其 他 文件 组 上 创建 数据 表 
前 面 的 示例 2 一 示例 4 所 创建 的 数据 表 全 部 都 存放 在 了 chapter3 数据 库 中 的 主 文件 组 
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中 。 在 第 2 章 学 习 数据 库 的 操作 时 就 提 到 过 ， 在 一 个 数据 库 中 可 以 有 多 个 文件 组 ， 但 是 只 
有 一 个 主 文件 组 ， 默 认 情况 下 数据 文件 都 会 存放 到 主 文件 组 中 ， 但 是 也 可 以 指定 文件 存放 
到 其 他 文件 组 中 。 不 仅 是 数据 文件 ， 数 据 表 也 是 可 以 指定 其 存放 的 文件 组 的 。 具 体 的 语句 


如 下 


CREATE TABLE table _ name 


( 


columnl name datatype, 
column2 name datatype, 


) 


ON filegroup name; 


这 里 ，filegroup_name 就 是 文件 组 的 名 字 。 
【示例 S】 根据 表 3-5 所 示 的 字段 信息 创建 用 户 信息 表 (userinfo3)， 并 将 该 数据 表 创 
建 在 chapter3 数据 库 中 的 chapterfilegroup 文件 组 中 。 
根据 题目 要 求 ， 假 设 在 创建 chapter3 数据 库 时 ， 添 加 了 文件 组 chapterfilegroup。 具 体 


的 语句 如 下 : 


USE chapter3 


-- 打 开 chapter3 数据 库 


CREATE TABLE userinfo3 


( 


id int, 


name varchar(20), 
password varchar (10), 
email Varchar (20) ， 
ee] varchar (15) ， 
el varchar (15) 


) 


ON chapterfilegroup; 


执行 上 面 的 语句 ,就 可 以 在 chapter3 数据 库 中 的 chapterfilegroup 文件 组 里 创建 userinfo3 
数据 表 。 执 行 效 果 如 图 3.4 所 示 。 


文件 外 编辑 于 ) 视图 WW) 查询 @) 项 目 E) 调试 @) 工具 C) 窗口 W) 社区 人 C) 


=|Glx| 


帮助 0 

辽 新建 查询 W | 由 轧 也 下 记 | 七 回 己 | 怠 天 

时 让 | chapte3 -| ?# 生 W 》 m v 光量 加 | 下 弘 | 的 图 名 
SQLQueryl. s... ator (53))*| vx 
日 UsE chapter3 一 打开 chapcer3 煞 据 库 


CREATE TABLE userinfo3 


id int 
name 


password varchar (10), 


email 


tel 


-ON chapterfilegroup; - 
局 消息 | 
命令 已 成 功 完成 。 = 
MA :型 


varchar [20) , 


varchar (20), 
varchar (15), 
varchar 115 


园 查 | YIDw-9527WSSQLSERVER2008 ( .| WIDW-9527\Adninistrato..，| chspter3 |00:00:00 |0 行 


行 1 列 17 Ch9 Ins /| 


图 3.4 创建 表 userinfo3 
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3.2.6 见识 一 下 临时 表 


所 谓 临 时 表 , 就 不 是 数据 库 中 永久 存在 的 表 。 临时 表 又 分 为 本 地 临时 表 和 全 局 临时 表 。 
本 地 临时 表 是 以 “#” 开 头 的 数据 表 ， 在 当前 用 户 下 可 用 ; 全 局 临时 表 是 以 “ 老 ” 开 头 的 数 
据 表 ， 所 有 用 户 都 可 以 使 用 。 临 时 表 就 好 像 是 超市 的 购物 车 ， 当 购物 的 时 候 需 要 ， 结 账 后 
就 不 再 需要 了 。 临 时 表 的 创建 语法 与 一 般 的 数据 表 创 建 是 一 样 的 ， 只 是 临时 表 通 常 都 存放 
在 tempdb 数据 库 中 。 另 外 ， 就 是 临时 表 的 名 字 都 要 以 “#” 或 “二 ” 作 为 前 组 。 

【示例 6】 创建 一 个 临时 表 (#usertemp1)， 表 中 的 字段 信息 如 表 3-6 所 示 。 


表 3-6 用 户 信息 临时 表 (#usertemp1) 


数据 类 型 
int 
varchar(20) 


根据 题目 要 求 ， 创 建 临时 表 usertemp1l 的 语句 如 下 : 


CREATE TABLE #usertempl 
( 
a dine 
name varchar (20) ， 
password varchar (10) 


执行 上 面 的 语句 后 , 就 可 以 在 tempdb 数据 库 中 创建 一 个 名 为 加 sertemp1 的 临时 表 。 执 
行 效果 如 图 3.5 所 示 。 和 


<Microso erver t Studi 
全 注意 ; 虽然 当前 打开 的 数据 库 是 chapter3， 但 是 临 ”国生 和 站 


时 表 依旧 会 创建 在 tempdb 数据 库 中 。 志和 本 
Re i 


CREATE TABLE 


3.2.7 ”使 用 企业 管理 器 轻松 创建 数据 表 


创建 数据 表 需 要 记 住 这 么 多 的 语法 真是 麻烦 啊 ， 
相信 读者 也 已 经 厌烦 了 吧 ? 打 起 精神 吧 ， 现 在 告诉 你 
一 个 简单 的 方法 创建 数据 表 ， 既 不 用 担心 忘记 数据 类 图 一 
型 的 名 称 又 不 用 怕 记 不 住 语法 ， 这 个 方法 就 是 使 用 企 ee ia eol 才 
业 管 理 器 。 显 示 企 业 管理 器 威力 的 时 间 到 了 ， 把 使 用 图 3.5 创建 临时 表 #usertemp1 
SQL 语句 创建 用 户 信息 表 的 过 程 在 企业 管理 器 中 通 
过 示例 7 演练 演练 。 

【示例 7】 在 企业 管理 器 中 ， 根 据 表 3-5 所 示 的 用 户 信息 表 的 字段 信息 分 别 按 如 下 两 
个 要 求 创建 用 户 信 息 表 userinfo4 和 userinfo5。 完 成 如 下 的 设置 。 

(1) 设置 用 户 信息 表 中 的 编号 列 为 自动 增长 字段 。 

(2) 使 用 用 户 自 定义 的 数据 类 型 usertype。 

根据 题目 的 要 求 ， 要 在 企业 管理 器 中 创建 数据 表 ， 在 本 例 中 仍然 将 表 创建 在 数据 库 


| 已 消息 | 
命令 已 成 功 完成 
于 
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chapter3 中 。 无 论 创建 的 数据 表 有 什么 要 求 ， 都 需要 在 表 的 设计 页 面 中 来 完成 。 下 面 就 先 
来 见识 一 下 表 的 设计 页 面 。 非 常 简单 哦 ， 只 需要 在 SQL Server 企业 管理 器 的 对 象 资源 管理 
器 中 找到 chapter3 数据 库 并 展开 文件 夹 ， 然 后 再 右 击 其 中 的 “ 表 ” 节 点 ， 在 弹出 的 右键 菜 
单 中 选择 “新 建 表 ” 选 项 ， 出 现 如 图 3.6 所 示 界 面 。 


EEC ln 局 6 可 
ET 编辑 到) ”视图 WD 项目) WD 甫 设计 器 LL) 工具 中 窗口 如 社区 人 帮助 0 
Wesaw | 中 | 书包 名 DS 日 G | 台电 


ovIDY-9527\g. .. dbo. Table_1 


图 3.6 表 设计 器 界面 


图 3.6 所 示 的 界面 就 是 创建 数据 表 的 操作 界面 ， 被 称 为 数据 表 的 设计 界面 。 所 有 关于 
数据 表 的 操作 都 要 在 该 界面 中 完成 。 下 面 就 使 用 该 界面 分 别 完成 本 题 的 两 个 小 题 。 


1. 自动 增长 字段 


根据 题目 的 要 求 ， 要 先 录入 用 户 信息 表 的 基本 内 容 ， 然 后 将 表 中 的 编号 列 为 自动 增长 
字段 。 完 成 这 个 要 求 要 分 为 如 下 3 个 步骤 。 

(1) 按照 表 3-5 的 要 求 ， 录 入 用 户 信息 表 的 信息 

在 图 3.6 所 示 的 界面 中 ， 录 入 用 户 信息 表 的 列 名 和 数据 类 型 ， 其 中 数据 类 型 是 可 以 通 
过 下 拉 列 表 选 择 的 。 录 入 后 效果 如 图 3.7 所 示 。 


na die 
文件 @) 编辑 EE) 视图 CD) 项目) 调式) 表 设 计 器 QL) 工具 〇 窗口 鸣 社区 C) 帮助 0 


varchar(20) 
varchar(10) 
varchar(20) 
varchar(15) 
Yarchar(15) 


图 3.7 录入 用 户 信息 表 信息 后 的 效果 
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在 图 3.7 中 ， 可 以 看 到 数据 类 型 后 面 还 有 一 列 “ 人 允许 Null 值 ”， 该 列 是 做 什么 的 呢 ? 
没 错 ， 正 如 字面 的 意思 就 是 设置 该 列 是 否 允 许 不 输入 值 的 ， 默 认 情况 下 ， 都 会 将 其 选中 即 
可 以 不 输入 值 的 。 这 种 是 否 为 空 的 限制 也 被 称 为 非 空 约束 。 关 于 非 空 约束 的 定义 将 在 第 4 
章 中 详细 讲解 。 

(2) 设置 编号 (id) 列 为 自动 增长 

设置 某 一 个 列 为 自动 增长 列 时 ， 前 提 是 该 列 的 数据 类 型 是 整数 。 在 企业 管理 器 中 设置 
自动 增 列 是 很 容易 的 ， 只 需要 在 列 的 属性 界面 中 完成 即 可 。 在 图 3.7 所 示 界 面 中 ， 单 击 id 
所 在 的 行 ， 找 到 id 的 列 属性 ， 如 图 3.8 所 示 。 


”lolx 
文件 四 ”编辑 @ 视图 WD。 项目 四) 调 坛 @) 表 设 计 器 UL) 工具 上 窗口 轨 社区 C) 帮助 


全 和 寻 查询 中 | 让 | 孔 也 怠 | 户 | 区 加 局 | 虽 中 
| 

EE | 
日 数据 库 varchar(20) 
varchar(10) 
varchar(20) 


varchar(15) 
varchar(15) 


VIDY-9527\0. .. dbo. Table_1* 


日 国 aepter4 
田 国 数据 库 关系 图 
田 筷 这 
图 国 视图 
图 国 同义词 
田 国 可 编程 性 
园 国 Service Brol 
困 国 存储 
田园 安全 性 
® ReportServer 
图 ReportServer 


图 国 安全 性 
田 国 服务 器 对 象 


1 
就 绪 


图 3.8 id 的 列 属性 界面 


在 图 3.8 所 示 界 面 中 ， 标 识 规范 选项 就 是 用 来 设置 自动 增长 的 ， 将 其 改 成 “是 ” 即 可 
将 该 列 设置 成 自动 增长 的 。 如 果 不 加 其 他 的 设置 ， 设 置 的 自动 增长 列 就 是 从 1 开始 每 次 增 
加 1 的 效果 。 如 果 要 设置 自动 增长 的 起 始 值 以 及 增 量 ， 如 在 本 例 中 将 其 起 始 值 设置 成 1， 
增 量 设置 成 10， 效 果 如 图 3.9 所 示 。 

在 图 3.9 中 ,“ 标 识 增 量 ”选项 就 是 要 设置 的 增 量 ,“ 标 识 种 子 ” 选 项 就 是 初始 值 。 读 
者 会 发 现 ， 在 将 id 设置 成 自 增长 的 列 后 ， 该 列 中 的 “允许 Null 值 ” 列 就 去 除了 选中 状态 ， 
也 就 是 不 允许 为 空 了 。 没 错 ， 设 置 了 自动 增长 值 后 是 不 会 产生 空 值 的 。 

(3) 给 表 命 名 

在 完成 了 表 的 信息 添加 和 自动 增长 列 设置 后 ， 一 定 不 要 忘记 保存 表 的 信息 哦 ! 保存 表 
的 信息 方法 有 很 多 , 这 里 介绍 几 个 常用 的 方法 吧 。 最 简单 的 方法 是 像 保 存 文件 一 样 用 Ctrl+S 
组 合 键 ， 不 愿意 用 键盘 还 可 以 使 用 工具 栏 上 的 加 按钮 来 保存 表 信 息 。 除 了 上 面 的 两 种 方法 
外 ， 还 可 以 使 用 文件 菜单 下 的 “保存 ”选项 来 完成 保存 的 操作 。 不 论 使 用 哪 种 方法 保存 表 
的 信息 ， 都 会 弹出 如 图 3.10 所 示 的 界面 。 
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宣 
日 国 数据 库 name varchar(20) a 
田 加 系 坟 数据 库 一 ss 所 
田 加 尖 所 库 志 
chapter2l emall varchar(20) 3 
田 国 chapter22 varchar(15) 3 
田 国 chapter23 tel varchar(15) 克 
日 是 hepter3 二 
田 国 xx ls | 
电导 列 时 性 
田 加 视图 | 
田 国 同义词 贺 色 豆 
田 加 可 编程 性 El . 
田 国 Serviee Drok 
田 国 存储 全 口 
四 国安 全 性 二 妆 二 下 
图 国 chapter4 数据 基 型 nt 
田 国 ReportserverSnssqLsERyeR2(| | 区 放 NA 什 天 
团 日 ReportServer$NSSQLSERYER2( | | 日 表 设计 器 
田间 安全 性 RowGud 香 
田 加 服务 器 对 象 回 是 
加 加 复制 (是 标识 ) 是 
田 襄 管理 标识 增 量 10 
加 SQL server 代理 已 禁用 代理 并 标识 种 子 1 
2 在 习 EH x 
到 


图 3.9 设置 自动 增长 列 图 3.10 保存 表 信息 界面 


在 图 3.10 所 示 界 面 中 ， 输 入 表 的 名 称 ， 按 照 本 题 的 要 求 输入 “userinfo4”， 单 击 “ 保 
存 ” 按 钮 , 即 可 完成 userinfo4 的 创建 操作 。 还 有 一 点 需要 注意 哦 , 那 就 是 名 字 不 要 与 chaper3 
中 的 数据 表 重 名 ! 完 成 保存 操作 后 , 在 chapter3 数据 库 中 的 “ 表 ” 节 点 下 , 就 会 出 现 userinfo4 
表 的 名 字 了 ， 如 图 3.11 所 示 。 


2. 用 户 自 定义 的 数据 类 型 


在 设置 自 定义 数据 类 型 之 前 ， 不 要 忘记 先 创 建 自 定义 数据 类 型 。 在 本 题 中 就 先 创建 一 
个 用 户 自 定义 的 数据 类 型 ， 然 后 再 使 用 该 数据 类 型 。 具 体操 作 分 为 如 下 两 个 步骤 完成 。 

(1) 创建 自 定义 数据 类 型 usertype 

创建 自 定义 数据 类 型 前 先 要 找到 创建 用 户 定义 数据 类 型 的 位 置 ， 它 就 在 chapter3 数据 
库 中 的 “可 编程 性 ”节点 的 “类 型 ”节点 里 ， 如 图 3.12 所 示 。 


日 国 hwpters 


日 国 :hseter3 
田 国 数据 库 关 系 图 


田 国 Service Broker 
国 存 嵌 
国 安全 性 


图 3.11 表 节 点 下 的 userinfo4 图 3.12 用 户 定义 数据 类 型 的 位 置 
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在 图 3.12 所 示 界 面 中 , 右 击 “ 用 户 定义 数据 类 型 ”选项 , 在 弹出 的 右键 菜单 中 选择 “新 
建 用 户 定义 数据 类 型 ”选项 ， 弹 出 如 图 3.13 所 示 界 面 。 


3.13 ”新 建 用 户 定义 数据 类 型 界面 


在 图 3.13 所 示 的 界面 中 , 输入 自 定义 数据 类 型 的 名 称 , 然后 选择 一 个 数据 类 型 并 输入 
该 数据 类 型 的 长 度 。 这 里 ， 自 定义 数据 类 型 的 名 称 是 usertype， 数 据 类 型 是 varchar， 长 度 
是 20。 输 入 后 的 效果 如 图 3.14 所 示 。 


箱 洒 建 用 户 定义 数据 关 型 


图 3.14 输入 自 定义 值 后 的 效果 
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在 图 3.14 所 示 界 面 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 完成 自 定义 数据 类 型 的 添加 操作 。 

(2) 在 表 设 计 器 中 使 用 自 定义 数据 类 型 

设置 好 自 定义 数据 类 型 后 ， 在 使 用 该 数据 类 型 时 与 系统 的 数据 类 型 是 一 样 的 。 自 定义 
的 数据 类 型 也 会 出 现在 表 设 计 器 数据 类 型 的 下 拉 列 表 框 中 。 与 (1) 的 方法 一 样 ， 先 将 用 户 
信息 表 的 字段 信息 输入 表 设 计 器 中 ， 然 后 将 需要 使 用 varchar(20) 数 据 类 型 的 列 设置 成 用 户 


自 定义 数据 类 型 即 可 。 完 成 操作 后 ， 将 表 的 名 称 
保存 成 userinfo5。 具体 的 操作 与 (1) 的 方法 相同 ， 了 ELSE 


可 int 口 
请 读者 自己 尝试 模仿 一 下 吧 。 这 里 ， 为 了 确保 读 中 记 
者 能 够 找到 用 户 自 定义 数据 类 型 ， 将 表 设计 器 的 ” 当 字 
数据 类 型 显示 出 来 ， 如 图 3.15 所 示 。 “ 时 
从 图 3.15 可 以 看 出 ， 自 定义 的 数据 类 型 会 显 所 
示 在 数据 类 型 列表 的 最 后 面 。 To 


通过 示例 7 的 讲解 ， 相 信 读 者 已 经 发 现 了 使 图 3.15 自 定义 数据 类 型 显示 的 位 置 
用 企业 管 器 创建 数据 表 还 是 很 方便 的 吧 ! 但 是 ， 
读者 也 不 要 放松 对 SQL 语句 的 学 习 哦 ! 


3.2.8 使 用 SP_HELP 看 看 表 的 骨架 


数据 表 创 建 好 后 ， 如 何 查看 一 下 数据 表 呢 ? 当然 ， 用 企业 管理 器 就 可 以 看 容易 地 查看 
数据 了 。 如 果 不 使 用 企业 管理 器 如 何 查看 数据 表 的 信息 呢 ? 有 很 多 方法 可 以 查看 数据 表 的 
信息 ， 这 里 先 介绍 使 用 SP_HELP 存储 过 程 来 查看 数据 表 的 信息 。 查 看 的 语句 如 下 : 


SP_HELP table name; 


这 里 ，table_name 是 数据 表 的 名 称 。 在 查看 数据 表 之 前 ， 不 要 态 记 用 USE 语句 指定 要 
使 用 的 数据 库 哦 。 

【示例 8】 使 用 存储 过 程 SP_HELP 查看 用 户 信息 表 〈userinfo) 的 表 信息 。 

根据 题目 的 要 求 ， 查 看 的 语句 如 下 : 


SP_HELP userinfo; 


执行 上 面 的 语句 ， 效 果 如 图 3.16 所 示 。 

在 图 3.16 中 查询 结果 划分 成 了 5 个 部 分 ,下 面 分 别 说 明 这 5 部 分 显示 的 信息 都 是 什么 。 

口 第 1 部 分 ， 显 示 表 创建 时 的 基本 信息 ， 包 括 数据 表 的 名 称 、 类 型 和 创建 时 间 以 及 
拥有 者 。 

口 第 2 部 分 显示 表 中 列 的 信息 ， 包 括 列 的 名 称 、 数 据 类 型 和 长 度 等 信息 。 

口 第 3 部 分 : 显示 表 中 自动 增长 列 的 信息 。 由 于 在 表 userinfo 中 没有 自动 增长 列 ， 因 
此 在 本 图 中 没有 自动 增长 列 。 

口 第 4 部 分 : 显示 表 中 的 全 局 唯一 标识 符 列 。 在 每 一 个 数据 表 中 只 能 有 一 个 全 局 唯 
一 标识 符 列 。 在 表 userinfo 中 也 没有 设置 全 局 唯一 标识 符 列 。 

口 第 5 部 分 : 显示 表 存在 的 文件 组 。 在 本 图 中 显示 的 是 userinfo 存放 在 主 文件 组 
(PRIMARY) 中 。 
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文件 四 ”编辑 加 ”视图 WW) 查询 @) 项 目 E) 调试 四 工具 I) 鲁 口 WD 社区 人 帮助 0) 
ET 


Ee -|!#5w 》 mv 罗 名 国 | 洲 允 | 六 图 各 | 三 S| 幸 地 | 急电 
Sauery2 ator G2))*] 区 

Sp_HELP userinfo: 
器 结果 | 2 消息 | 

Name | Dwner [ Type | Diesed daletme 
[ema | ao。 veriable 20121109074533247 

on Type | Computed | Longh | Prec | Scale | Notable | TrmTraingdlonks | Fived erNulinSource | Colation 

mm 4 10 0 ye md al NULL 

name vacha mm a wm yes Chinese_PRC_ CL_AS 

password vacha m 10 pm yes Chinese_PRC_CLAS 

Emal vacha 四 a ys 四 yes Chinese_PRC._ OL_AS 

o0 vacha mo 本 pm yes Chinese_PRC_ OI_AS 

加 vachat ro 下 gsm yes Chinese PRC_CLAS 

Te Seed | nctement | Not For Rephcabo 
[Nedenty cobmn deined | NULL NULL NULL 
[Neowguaealcaumn deined 

区 

PRIMARY S| 
团 可 9 已 成功 执行 . | rrr-9527 nssoL SFRVERo008 (| TTD-9527\Adninistreto | chapter3 |o0.00:0! |10 行 
就 绪 行 1 列 18 ch 18 Ins 4 


图 3.16 ”userinfo 表 的 信息 


全 说 明 : 使 用 存储 过 程 SP_HELP 不 仅 可 以 查看 表 的 信息 ， 也 可 以 查看 数据 库 的 其 他 对 象 
以 及 用 户 自 定义 的 数据 类 型 等 信息 。 查 询 方 法 很 简单 ， 只 需要 执行 SP_HELP 存 
储 过 程 即 可 , 而 不 需要 再 添加 表 名 了 .。 直接 使 用 SP_HELP 查询 的 效果 , 如 图 3.17 
所 示 。 


文件 四 疯 辑 G) 视图 QD 查询 @) 项 目 E) 调式 @) 工具 CO 究 口 WD 社区 人 帮助 0D 

有 2D | 六 | 十 议 钙 | 访 | 区 回电 | 屋 避 

再 起 | hwtw3 了 |? 执行 0， 且 v 弹 昌国] 于 三 | 奴 国 站 ) 三 全 | 地 刘 | 避 晤 

‘SqLQuery2. 5... ator (52))*| -x 
SP_HELP 避 


~ | shspter3 |00:00:00 | 1965 行 
就 请 行 1 列 9 Che Ins /4 


图 3.17 SP_HELP 查询 的 效果 


在 图 3.17 的 查询 结果 中 有 两 个 部 分 组 成 , 一 部 分 用 于 显示 数据 库 chapter3 中 所 有 
的 数据 对 象 信息 ， 一 部 分 用 于 显示 数据 库 chapter3 中 自 定义 的 数据 类 型 信息 。 


3.2.9 使 用 sys.objects 查看 表 的 信息 


看 了 存储 过 程 SP_HELP 查看 的 结果 ， 读 者 觉得 有 点 混乱 吧 。 如 果 你 只 想 知道 数据 表 
的 创建 信息 ， 现 在 告诉 你 一 个 相对 简单 点 的 方法 ， 那 就 是 使 用 系统 表 sys.objects 来 查看 。 
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虽然 查看 的 结果 显示 清晰 , 但 是 使 用 的 SQL 语句 就 要 复杂 一 些 了 。 下 面 就 使 用 示例 9 来 演 
示 如 何 使 用 sys.objects 来 查看 表 的 信息 。 

【示例 9】 使 用 系统 表 sys.objects 查看 userinfo 表 的 信息 。 

根据 题目 的 要 求 ， 查 看 userinfo 表 的 信息 语句 如 下 : 


select * from sys.objects where name='userinfo' ; 


执行 上 面 的 语句 ， 就 可 以 将 userinfo 表 的 创建 信息 显示 出 来 。 效 果 如 图 3.18 所 示 。 


S ageeen = 
文件 四 编 得 @) 视 图 WD 查询 @) 项 目 F) 调式 W) 工具 0) 鲁 口 四 社区 CC) 帮助 o 
耳 建 查询 思 | 记 | 轧 耶 劝 巴 | 蕊 四 要 盈亏 
YH | chapter3 -| ?执行 有 v 史 可 国 六 泡 | 们 疾 训 | 三 全 | 这 这 | 人 hs 
SqlQuery2. s. .ator (2))*| -x 
select from Where name='userinfo' 


ncpaL yd | 20 


NUU 


团 查 i 已 成 功 执行 。 | YIDY-9527WISSQLSEBYEE2008 〔 .， WII-9527\Adninistrato.. .| chapter3 | 00:00:00 | 1 行 
就 绪 入 列 50 Ch 50 Ins 玫 


图 3.18 使 用 sys.objects 查看 userinfo 表 信 息 


从 图 3.18 的 查询 结果 中 ， 可 以 看 出 使 用 sys.objects 系统 表 可 以 查看 到 userinfo 表 的 创 
建 时 间 、 修 改 时 间 以 及 表 的 类 型 等 信息 。 学 习 了 查看 userinfo 表 的 信息 ， 相 信 读 者 已 经 看 
出 ， 在 sys.objects 中 的 name 列 就 是 要 查找 的 数据 对 象 名 称 ， 也 就 是 说 ， 要 查询 哪个 数据 
对 象 就 将 name 后 的 对 象 名 改 成 要 查找 的 对 象 即 可 。 当 然 ， 也 可 以 使 用 sys.objects 系统 表 
不 加 任何 条 件 查 看 数据 库 中 所 有 的 数据 对 象 。 


3.2.10 使 用 Information_schema.columns 查看 表 的 信息 


在 前 面 的 两 个 小 节 中 分 别 使 用 了 存储 过 程 和 系统 表 来 查看 表 的 信息 ， 除 了 这 两 种 方法 
外 , 还 可 以 使 用 系统 视图 Information_schema.columns 来 查看 表 的 信息 。 通 过 这 个 系统 视图 
可 以 查看 出 表 中 的 列 信息 , 但 是 不 包括 表 的 创建 信息 。 具 体 的 查看 方法 使 用 示例 10 来 告诉 
读者 。 

【示例 10】 使 用 Information _ schema.columns 查看 userinfo 表 的 信息 。 

根据 题目 要 求 ， 查 看 userinfo 表 的 信息 语句 如 下 : 


select * from Information schema.columns where table name='userinfo'; 


执行 上 面 的 语句 ， 效 果 如 图 3.19 所 示 。 
从 图 3.19 中 ,可 以 看 出 通过 Information_schema.columns 视图 可 以 查看 到 userinfo 表 所 
属 的 数据 库 名 、 列 名 以 及 列 的 数据 类 型 等 信息 。 


名 说明: 前 面 讲 过 的 使 用 SP_HELP、sys.objects 和 Information schema.columns 这 3 种 查 
询 表 信息 的 方式 ， 它 们 都 在 什么 时 候 使 用 呢 ? SP_HELP 主要 用 于 查询 表 中 所 有 
的 信息 ， 包 括 表 的 创建 信息 、 列 信息 以 及 其 他 的 信息 ; sys.objects 主要 用 于 查询 
表 的 创建 信息 ; Information schema.columns 用 于 查询 表 的 列 信息 。 知 道 了 它们 的 
用 途 ， 读 者 可 以 根据 自己 的 需要 自行 选择 了 吧 ! 
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目 中 调式 史 工具 襄 D 四 社区 CD 帮助 op 
也. 新建 查询 四 | 出 | 也 也 加 | 也 | 蕊 加 本 受 天 
了 | ?执行 四  Y 弹 避 [ 国 | 计 克 | 克 上 图 | 罗 | 三 全 | 地府 | 全 品 


G2))#| vx 
forma = where cable_name= userinto 

品 结果 | 二 消息 | 

TABLE_ CATALOG TABLE SCHEMA | TABLE_NAME | COLUMN_ NAME | ORDINAL POSITION | COUUMN_LDEFAULT | 15-NULLABLE | DATATYPE 
Te dbo senio 这 1 N YES 加 
2 | chape5 dho seinlo name z NULL YE5 varchar 
3 | chapter3 db sarin password NULL YES vachar 
| chapter dho sarinlo em 4 NULL YE5 vacha 
5 | chapter3 dho sem 虽 5 NULL Yes vachar 
| chapter dho sem a § NULL vES vacha 
uw 到 
回 本 已 成 功 执行 。 | rrp-9527 ssQLSEEVER2008 (| WTO-952T\Aninistrato ,| chapter3 |00:00:00 | 行 
行 1 Em ED ms 


图 3.19 使 用 Information_ schema.columns 查看 userinfo 表 


3.3 ”修改 数据 表 


任何 事务 都 不 可 避免 地 会 遇 到 修改 的 问题 ， 数 据 表 也 不 例外 。 创 建 好 的 数据 表 都 可 以 
修改 它 的 哪些 内 容 呢 ? 告诉 你 吧 ， 其 实 几 乎 所 有 的 内 容 都 是 可 以 修改 的 ， 包 括 修改 字段 的 
数据 类 型 、 添 加 或 减少 表 中 的 字段 、 修 改 表 中 字段 名 字 以 及 给 表 改名 等 。 另 外 ， 还 有 一 个 
好 工具 可 以 帮助 我 们 修改 数据 表 ， 那 就 是 企业 管理 器 。 在 本 节 中 ， 将 为 读者 详细 讲述 如 何 
修改 数据 表 。 

3.3.1 改 一 改 表 中 的 数据 类 型 

表 中 的 数据 类 型 ， 可 以 说 是 最 常 修改 的 一 项 内 容 了 。 之 所 以 说 它 是 最 常 修改 的 ， 其 实 
多 半 是 由 于 数据 表 的 设计 人 员 在 设计 表 时 设计 的 长 度 不 够 或 者 是 创建 表 时 写 错 了 。 修 改 数 
据 类 型 是 一 件 很 容易 的 事情 哦 ， 请 看 下 面 的 语法 吧 。 


ALTER TABLE table name 
ALTER COLUMN column name datatype; 


其 中 : 

口 table name: 表 名 。 要 修改 的 数据 表 名 ， 在 执行 修改 语句 时 ， 也 要 先 使 用 USE 语 
句 打开 表 所 在 的 数据 库 。 

口 column_name: 列 名 。 数 据 表 中 的 列 名 。 如 果 不 清楚 表 中 的 列 名 ， 可 以 先 查 看 一 下 
表 的 信息 再 进行 修改 。 


口 datatype: 数据 类 型 。 给 表 中 列 新 设置 的 数据 类 型 。 能 够 设置 新 类 型 的 前 提 是 该 字 
段 中 存放 的 值 能 够 兼容 新 设置 的 类 型 。 通 常 ， 都 是 在 表 中 还 没有 存放 数据 时 修改 
数据 类 型 。 
【示例 11】 修改 用 户 信息 表 (userinfo)， 将 其 中 的 用 户 名 列 Cname) 的 数据 类 型 改 成 
varchar(30)。 
根据 题目 要 求 ， 只 是 将 数据 类 型 的 长 度 更 改 了 一 下 ， 具 体 的 语句 如 下 : 
USE chapter3 -- 打 开 chapter3 数据 库 


ALTER TABLE userinfo 
ALTER COLUMN name varchar (30); 
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执行 上 面 的 语句 后 ， 就 可 以 将 用 户 信息 表 (userinfo) 中 的 姓名 列 Cname) 的 数据 类 型 


更 改 成 varchar(30) 了 。 执 行 效果 如 图 3.20 所 示 。 


大助 09 
卫 间 建 查询 中 | 六 | 六 六 名 | 访 | 访 回 时 又 瑟 
chapter -| 1 执 和 SW ha v 弹 于 回 | 了 码 | 妈 图 


-所 开 chapcer3 要 所 库 
由 ALTER TABLE 
上 ALTER COLUNN 


局 请 息 | 

命令 已 成 功 完成 。 
j 司 | 
园 坦 | rrssz7WssgLsEpygiz008 (| WIDW-9527\Adninistrato. .| chspter3 |00:00:00 |0 行 


就绪 行 1 列 45 Ch 40 


Ins 去 


图 3.20 ”更改 name 列 的 长 度 


3.3.2 更改 表 中 字段 的 数目 


如 果 一 张 数 据 表 创建 好 了 ， 根 据 实 际 的 项 目 需求 需要 添加 或 删除 一 些 字段 ， 这 时 就 可 


以 用 到 修改 表 中 字段 数目 的 语 名 了。 修改 表 中 字段 数目 的 语句 比较 简单 ， 但 是 也 得 读者 认 
真 记忆 ， 以 免 混 淆 。 


(1) 向 表 中 添加 字段 

ALTER TABLE table _ name 

ADD column name datatype; 

其 中 ， 

口 table name: 表 名 。 

口 column_name: 新 添加 的 列 名 。 列 名 不 能 与 表 中 已 经 存在 的 列 重 名 ， 因 此 在 添加 列 
时 最 好 先 查 看 一 下 表 中 现 有 列 的 信息 。 

口 datatype: 数据 类 型 。 

(2) 删除 表 中 的 字段 信息 

ALTER TABLE table name 

DROP COLUMN column name; 

其 中 ， 

口 table name: 表 名 。 

口 column name: 列 名 。 删 除 后 的 字段 可 不 能 恢复 了 ， 在 删除 前 可 要 考虑 清楚 了 。 

【示例 12】 向 用 户 信息 表 〈userinfo ) 中 添加 一 个 备注 列 (remark)， 数 据 类 型 是 


varchar(50)。 


名 。 


根据 题目 要 求 ， 查 看 了 用 户 信 息 表 后 ， 发 现 备 注 列 〈remark) 在 用 户 信息 表 中 没有 重 
具体 的 语句 如 下 : 
USE chapter3; 


ALTER TABLE userinfo 
ADD remark varchar(50); 


执行 上 面 的 语句 后 ， 就 在 用 户 信息 表 userinfo 中 就 增加 了 备注 列 (remark)。 执 行 效果 
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如 图 3.21 所 示 。 
四 
者 助 中 
卫 新建 查询 思 | 记 | 蕊 也 怠 让 | 七 四 己 | 又 所 
于 Ha | setera -| ?执行 D_ 》 mv 引 王国 | 于 码 | 妈 轿 


SQLQuery2. s... ator (52))*| 
日 USE chapter3; 
由 ALTER TABLE userinfo 


ADD remark varchar (50); 


局 消 息 | 
完 = 
ee 二 


加 可 WIDt-9527\SSQLSERYER2008 ( .| WIDW-9527\Adninistrato. .，| chapter3 |00:00:00 |0 行 
就 绪 行 3 列 27 Ch27 Ins 


图 3.21 添加 remark 列 


NN 


【示例 13】 删除 在 示例 12 中 为 用 户 信 息 表 (userinfo) 添加 的 备注 列 (remark)。 
根据 题目 的 要 求 ， 删 除 字段 的 语句 如 下 : 
USE chapter3; 


ALTER TABLE userinfo 
DROP COLUMN remark; 


执行 上 面 的 语句 ， 就 可 以 将 用 户 信 息 表 (userinfo〉 中 的 备注 列 (remark) 删除 了 。 执 
行 效果 如 图 3.22 所 示 。 


EEC olz 
文件 中 编辑 EE) 视图 WW) 查询 @) 项目) 调式 中 工具 CD) 窗口 WD 社区 C) 
帮助 0D 

卫 . 新 建 查询 | 记 | 也 也 本 | 记 | 区 回避 后 

肢 骸 | chepters -| 7356m 》 vv 中 避 回 守 盐 | 她 国 
idery2 sator (2))*| -x 
站 USE chapcer3: 四 
斩 ALTER TABLE userinfo 


| DROP COLUNN remark; 


园 查 | WIDw-9s27\ssQLSERVER2008 (| ITDY-9527Wdninistrate .|chspter3 |00:00:00 |0 行 


就 绪 行 3 列 23 Ch23 Ins 


图 3.22 ”删除 remark 列 


3.3.3 ”给 表 中 的 字段 改名 

要 想 给 表 中 的 字段 改名 ， 用 ALTER 语句 可 就 不 行 唆 。 那 用 什么 语句 来 给 表 中 的 字段 
改名 呢 ? 如 果 读 者 已 经 学 习 过 了 第 2 章 ,在 第 2 章 中 修改 数据 库 的 名 字 用 什么 来 着 ? 没 错 ， 
就 是 用 存储 过 程 sp_rename 来 完成 的 。 这 里 ， 也 可 以 通过 sp_rename 来 修改 字段 的 名 字 。 
具体 的 语法 如 下 : 

SP_rename "tablename .columnname '，'new_columnname "7 

其 中 

口 tablename.columnname: 原来 表 中 的 字段 名 。 记 住 ， 表 中 的 字段 名 要 加 上 单 引号 。 
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口 new_columnname: 新 字段 名 。 新 字段 名 也 要 加 上 单 引 号 ， 并 且 不 能 与 其 他 的 字段 
名 重 名 。 
【示例 14】 将 用 户 信息 表 (userinfo) 中 的 用 户 名 (name) 字段 的 名 字 更 改 成 username 。 
根据 题目 的 要 求 ， 对 用 户 名 字段 改名 ，username 在 用 户 信息 表 中 没有 与 之 重复 的 。 修 
改 语句 如 下 : 


sp_rename 'userinfo.name','username'; 


执行 上 面 的 语句 ， 就 可 以 将 用 户 信 息 表 (userinfo) 中 的 用 户 名 (name) 字段 的 名 字 改 
成 username 了 了。 执行 效果 如 图 3.23 所 示 。 


也 间 寻 查询 六 | 鞍 逻 鸟 | 局 | 区 回 当 | 如 5 
Pe "| vj 


站 sp_rename ‘userinfo.name','username' 


ee 的 任 一 部 分 都 可 能 会 破坏 脚本 和 存储 过 程 。 
副 

园 坦 | iT-sszyWssqLsFRyEE2006 (| ITDY-9527Wdaninistrate .. | chapter3 |00:00:00 |0 行 
就 绪 行 3 列 1 Chl Ins 4 


图 3.23 ”将 列 名 name 更 改 成 username 


从 图 3.23 中 ， 虽 然 没 有 看 到 “执行 成 功 ”的 字样 ， 但 是 列 名 已 经 更 改 了 。 读 者 可 以 利 


用 select*from Information_schema.columns where table_name='userinfo' 语 句 查 看 一 下 效果 。 


3.3.4 给 数据 表 也 改 个 名 

既然 表 中 的 字段 名 可 以 更 改 , 数据 表 的 名 字 能 改 吗 ? 答案 是 肯定 , 但 是 不 能 用 ALTER 
语句 修改 。 也 要 用 sp_rename 语句 来 修改 ， 这 回 修改 语句 可 与 修改 数据 库 名 字 的 语句 类 似 
了 。 把 你 想到 的 和 下 面 的 语法 对 照 一 下 ， 看 看 是 不 是 想 对 了 呢 ? 

SP_rename old tablename,new tablename; 

这 里 ，old tablename 是 原来 的 表 名 ，new_tablename 是 修改 后 的 表 名 。 在 更 改 表 名 的 
时 候 也 要 记得 留意 一 下 数据 库 中 是 和 否 有 你 要 创建 的 表 名 。 另 外 ， 在 修改 表 名 之 前 还 要 将 该 
表 存 在 的 数据 库 用 USE 打开 。 

【示例 15】 将 用 户 信息 表 (userinfo) 的 名 字 修 改 成 Cnewuserinfo ) 。 

根据 题目 要 求 ， 新 表 名 newuserinfo 在 chapter3 数据 库 中 不 存在 。 修 改 的 语句 如 下 : 


sp_rename userinfo,newuserinfo; 


执行 上 面 的 语句 ， 在 chapter3 中 就 没有 名 为 userinfo 的 数据 表 了 ， 而 多 了 一 个 名 字 为 
newuserinfo 的 数据 表 。 执 行 效 果 如 图 3.24 所 示 。 


3.3.5 ”使 用 企业 管理 器 更 容易 修改 表 
有 了 在 企业 管理 器 中 创建 表 的 基础 ， 使 用 企业 管理 器 修改 数据 表 就 很 容易 了 。 下 面 就 
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文件 旭 ”编辑 人 视图 WD 查询 @) 
帮助 


了 利 寻 查询 | 出 | 也 也 可 | 也 
BY 起 | ceotes 


-| 执行 Ww av 允 可 国政 的 | 六 赔 癌 


走 进 SQL Server 


项 目 和 Ey) 调试 @) 工具 CI 窗 DWD 社区 CD) 


ETET 


‘SQLQuery2. s... ator (52))*| 
mi 


已 消息 | 
注意 : 


| 


| 日 sp_rename userinfo,newuserinto. 


更 改 对 象 名 的 任 一 部 分 都 可 能 会 破坏 脚本 和 存储 过 程 . 


回音 | mr-sszTwssQLsERyEF2008 (| ITDY-952TWAaninistrate .| chapter3 |00:00:00 |0 行 


行 1 


列 36 Ch35 


图 3.24 将 表 userinfo 更 改 成 newuserinfo 


用 示例 16 演示 一 下 如 何在 企业 管理 器 中 修改 数据 表 的 操作 。 读 者 也 可 以 根据 自己 的 想法 尝 


试 修改 一 下 ， 然 后 对 照 示例 16 检查 一 下 操作 是 否 正确 。 
【示例 16】 在 企业 管理 器 中 ， 按 照 如 下 要 求 修 改 用 户 信息 表 userinfo。 
(1) 向 用 户 信息 表 添 加 备注 字段 (remark)， 数 据 类 型 是 varchar(50) 
(2) 将 备注 字段 (remark) 的 数据 类 型 修改 成 varchar(100) 
(3) 将 备注 字段 (remark) 重 命名 成 remarks 


(4) 删除 备注 字段 (remarks) 


根据 题目 的 要 求 ， 这 4 个 小 题 都 将 在 企业 管理 器 中 用 户 信息 表 (userinfo) 的 表 设 计 界 
面 中 完成 。 那 么 ， 就 先 见识 一 下 修改 表 时 的 表 设 计 界面 吧 。 在 企业 管理 器 中 的 对 象 资源 管 
理 器 里 ， 展 开 chapter3 数据 库 节 点 ， 在 其 中 的 “ 表 ” 节 点 下 ， 右 击 userinfo 表 ， 在 弹出 的 


右键 菜单 中 选择 “设计 ”选项 ， 如 图 3.25 所 示 。 
下 面 就 在 图 3.25 所 示 的 界面 中 ， 按 照 题目 的 顺序 演示 如 何 修改 数据 表 。 


(1) 添加 字段 操作 很 简单 哦 ， 只 要 在 图 3.25 所 示 的 界面 中 ，tel 所 在 行 的 下 方 ， 单 击 
“ 列 名 ”所 对 应 的 单元 格 ， 并 录入 列 名 remark， 数据 类 型 为 varchar(50) 即 可 。 录 入 后 的 效果 


如 图 3.26 所 示 。 


ver Nanagement Studio 
文件 人 编辑 E) 视图 OO ”项目 全) 调试 @) 表 设计 器 必 ) 
工具 C) 窗口 WD 社区 C) 帮助 00 


卫 新建 查询 mg | 由 | 也 阴 加 | 记 | 记 局 包 | 运 天 
SOE EETFER 


| TIDY-9527W... dbo. aserinfo x 
ne ng 


EE] ink [Le 
Username varchar(30) 区 
password archar(10) a 
emall archar(20) 区 
qq varchar(15) [3 
tel varchar(15) 区 


文件 也 编辑 下 ) 视图 QD) 


项 目 @) 调 坛 @) 表 设 计 器 必 ) 
工具 C) 从 Do 社区 中 帮助 0 


卫 新 建 查询 W | 出 也 也 加 | 出 | 记 四 外 | 总 瑟 
轧 | 了 123 间 讽 讽 回回 司 


站 nt 


Usermame Yarchar(30) 
password varchar(10) 
emal varchar(20) 
QQ varchar(15) 


tel varchar(15) 
Yarchar(50) 


习习 习习 习习] 


VIDY-9527\0. .. 0. userinfos* vx 
列 名 数据 类型 交 许 Nul 值 


图 3.25 ”userinfo 表 的 设计 界面 


已 保存 的 项 


图 3.26 添加 字段 remark 的 效果 


添加 字段 后 ， 记 得 要 保存 表 的 信息 。 


(2) 修改 remark 字段 的 数据 类 型 在 图 3.26 所 示 的 界面 中 操作 就 可 以 了 , 直接 将 remark 
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字段 的 数据 类 型 varchar(50) 改 成 varchar(100)。 效 果 如 图 3.27 所 示 。 

同样 ， 也 要 将 图 3.27 所 示 的 结果 保存 后 才 完成 修改 操作 哦 ! 

(3) 重 命名 remark 字段 ， 与 修改 remark 字段 的 数据 类 型 相似 ， 在 图 3.27 所 示 的 界面 
中 操作 就 可 以 了 。 操 作 方 法 就 是 直接 单 击 remark 所 在 的 单元 格 ， 然 后 将 其 改 成 remarks 即 
可 。 效 果 如 图 3.28 所 示 。 

修改 remark 字段 的 名 称 后 ， 保 存 即 可 。 

(4) 删除 remarks 字段 在 图 3.28 所 示 的 界面 中 操作 就 可 以 ， 右 击 remarks 字段 所 在 的 
行 ， 在 弹出 的 右键 菜单 中 选择 “删除 列 ”选项 ， 即 可 将 该 列 删 除 。 删 除 后 也 不 要 忘记 保存 
表 信息 ! 


=|Gl 寺 | x| 
文件 旭 篇 加 于 ) 视图 WD 项目 下) 调式 呈 ) 表 设计 器 
工具 中 窗口 如 社区 C) 帮助 0 工具 WD 定 DWD 区 局 箱 助 
也. 章 奸 查询 W 六 六 六 芒 中 | 臣 回 和 妥 s 巡 新 尘 查 油 人 D 启动 六 入 忆 | 马 回 局 岗 忆 
最 | ?| 芭 加 商 测 回 加 训 站 
| YIDY-9527\. - -。 aserinfo# vx | YIDY-9527\g. - -。 userinfo# vx 
到 名 小 据 关 型 | 区 许 Al 值 | 列 数据 关 型 | 区 评 Nul 值 
] int [a EE] int [=] 
Username Varchar(30) a varchar(30) La 
password varchar(10) Ea varchar(10) 反 
emal varchar(20) [a varchar(20) a 
QQ varchar(15) Ea varchar(15) 区 
tel varchar(15) [3 varchar(15) 区 
remark 区 varchar(100) 区 
[| LU | 下 | 
列 飞 性 | 3 性 | 
辆 外 时 | 贺 引 | 刁 | 
已 保存 的 项 网 已 保存 的 项 4 
图 3.27 修改 remark 字段 的 数据 类 型 图 3.28 重 命名 remark 字段 


外 说 明 : 在 示例 16 中 讲解 了 在 企业 管理 器 中 修改 表 的 一 些 操作 ， 除 了 这 些 操作 外 ， 还 可 
以 进行 将 表 中 的 列 删除 、 给 列 设 置 成 标识 列 等 操作 。 


3.4 删除 数据 表 


当 不 再 需要 某 一 个 数据 表 时 ， 可 以 通过 SQL 语句 或 者 是 企业 管理 器 将 其 删除 。 但 是 ， 
删除 后 的 数据 表 就 很 难 恢复 了 。 因 此 ， 在 删除 数据 表 前 一 定 要 先 将 数据 表 备 份 一 下 ， 以 免 
带 来 不 必要 的 损失 哦 ! 


3.4.1 删除 数据 表 的 语 ; 


删除 数据 表 的 语法 要 比 前 面 介绍 的 创建 、 修 改 数据 表 的 语法 简单 多 了 ， 记 住 DROP 关 
键 字 就 差不多 了 ! 创建 和 修改 数据 表 时 每 次 只 能 创建 或 修改 一 张 数 据 表 ， 但 是 删除 的 时 候 
则 可 以 一 次 删除 多 张 数据 表 ! 这 就 像 盖 房子 一 样 ， 房 子 得 一 层 一 层 地 盖 ， 但 是 拆 房子 却 可 
以 通过 爆破 一 下 把 整个 房子 都 拆 了 。 拆 了 很 难 恢复 了 ， 读 者 一 定 要 小 心地 对 待 删除 表 的 操 
作 啊 ! 删除 数据 表 的 语法 如 下 : 


DROP TABLE database_name.table_name1l，database_name.table_name2，… 
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其 中 : 

口 database name: 数据 库 名 。 要 删除 的 表 所 在 的 数据 库 名 。 如 果 已 经 把 表 所 在 的 数 
据 库 打开 了 ， 那 么 ， 数 据 库 名 就 可 以 省 略 了 。 但 是 ， 要 删除 其 他 数据 库 中 的 表 ， 
就 要 加 上 数据 库 名 。 

口 table name: 表 名 。 


3.4.2 使 用 DROP 语句 去 掉 多 余 的 表 


通过 学 习 DROP TABLE 删除 数据 表 的 语句 ， 读 者 已 经 清楚 如 何 删除 数据 表 了 吧 ! 下 
面 就 通过 下 面 儿 个 例子 来 应 用 一 下 DROP TABLE 语句 。 

【示例 17】 删除 chapter3 数据 库 中 的 userinfo1。 

根据 题目 要 求 ， 要 删除 的 表 是 userinfol1。 删 除 语句 如 下 : 


USE chapter3 

DROP TABLE userinfol; 

执行 上 面 的 语句 , userinfol 表 就 从 数据 库 chapter3 中 移 除了 ,执行 效果 如 图 3.29 所 示 。 

该 示例 不 仅 可 以 使 用 上 面 的 语句 删除 表 userinfo1 ， 也 可 以 使 用 DROP TABLE 
chapter3.userinfol 语句 来 完成 。 

【示例 18】 同时 删除 chapter3 数据 库 中 的 userinfo2 和 userinfo3。 

根据 题目 的 要 求 ， 一 次 要 删除 两 张 数据 表 。 有 具体 的 语句 如 下 : 

USE chapter3; 

DROP TABLE userinfo2, userinfo3; 

执行 上 面 的 语句 ， 就 可 以 将 表 userinfo2 和 userinfo3 从 数据 库 chapter3 中 移 除 了 。 执 
行 效 果 如 图 3.30 所 示 。 


gicrosoft SQL Server Nansgeneat Stadio 
文件 中) 编辑 于) 视图 WD 查询 @) 项 目 E) 调式 @) 工具 CD) 
窗口 ) 社区 C) 帮助 如 
PT 

Wy Wa | chapter3 -| ?执行 O 》 则 池 闻 总 


SQLQuery3. s... ator (52))* 
USE chapter3 
LDROP TABLE userinfo 


文件 四， 编辑) 视图 WW) 查询 @) 项目) 调 坛 ) 工具 CD) 
定 口 人 D 社区 帮助 

有 新 查询 WD 出 声 也 四 | 轧 芒 回 对 | 册 电 

对 0 | chapter3 | 执行 C) bP 四 部 交 


SQL9uery3. s- - - ator (52))#* 
USE chapter3; 
DROP TABLE userinfo2，userin 


已 消息 | 
命令 已 成 功 完成 。 
| 

[SERVER2O08 (.. | WIDW-9527\Administrato. .. | chapter3 |00:00:00 |0 行 
识 绪 行 3 列 1 Ch 1 网 


SERVER2008 (.. | WIDY-9527\Adninistrato. 
就 绪 行 3 


| chapter3 |00:00:00 |0 行 
列 1 Chl 


图 3.29 删除 表 userinfol 图 3.30 删除 表 userinfo2 和 userinfo3 
全 注意 :如 果 要 删除 的 数据 表 在 数据 库 中 不 存在 ， 在 执行 删除 语句 后 ， 会 出 现 如 图 3.31 
所 示 的 错误 提示 。 


3.4.3 ”使 用 企业 管理 器 轻松 删除 表 


现在 就 来 讲解 在 企业 管理 器 中 最 简单 的 一 个 表 操 作 了 ， 那 就 是 删除 数据 表 。 在 企业 管 
理 器 中 删除 数据 表 时 不 需要 使 用 表 设 计 器 ,直接 使 用 鼠标 就 可 以 操作 了 。 下 面 就 用 示例 19 
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:也 新 建 查 询 p | 让 | 了 也 台所 | 蕊 加 局 | 怠 瑟 
; Wy Ha | cteu3 -| 执行 W) av 总 鱼 回 | 弘 | 葡 圈 2 
vx 


SQLQuery3. <... a ter (52))*| WIDN-9527\N .0 userinfo# | 
i USE chaprcer3: 
L DROP TABLE userinfol; 


消息 | 
让 3701， 级别 11， 状 态 5s， 第 z 

全 林村 ,中 各 和 六” 抽 不 存在 或 痢 您 没有 所 希 多 权限、 
让 


D3 | WT-9527 WSSQLSERVER2008 (.. | TIDY-9527WAaninistrate .| chapter3 |00:00:00 | 0 行 


图 3.31 删除 不 存在 表 的 错误 提示 


来 演示 删除 操作 。 

【示例 19】 在 企业 管理 器 中 删除 userinfo 表 。 

在 对 象 资源 管理 器 里 ， 右 击 要 删除 的 数据 表 userinfo， 在 弹出 的 右键 菜单 中 选择 “ 删 
除 ” 选 项 ， 弹 出 如 图 3.32 所 示 的 界面 。 


显示 休 鞍 关系 0 ， 
[EEC 
图 3.32 ”删除 表 界 面 
在 图 3.32 所 示 的 界面 中 ， 单 击 “ 确 定 ” 按 钮 即 可 将 userinfo 表 删 除了 。 
3.5 本 章 


在 本 章 中 主要 讲解 了 SQL Server 里 数据 表 字 段 的 数据 类 型 、 使 用 SQL 语句 和 企业 管 
理 器 创建 数据 表 、 修 改 数据 表 以 及 删除 数据 表 。 其 中 , 在 讲解 数据 类 型 时 着 重 讲 解 了 整 型 、 
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浮 点 型 以 及 字符 串 类 型 的 使 用 。 在 创建 数据 表 部 分 除了 讲解 数据 表 的 创建 ， 还 讲解 了 如 何 
使 用 存储 过 程 SP_HELP、 系 统 表 sys.objects 和 系统 视图 Information schema.columns 来 查 
看 数据 表 的 信息 。 最 后 ， 给 读者 提 个 醒 吧 ， 如 果 想 成 为 一 个 优秀 的 数据 库 管理 器 员 ， 还 是 
要 记 牢 SQL 语句 ， 不 能 单 靠 企业 管理 器 来 操作 的 。 


3.6 本 章 习 题 


一 、 填 空 题 


下 
2 


l; 


和 


加 


1 
2， 
2 


常见 的 数据 类 型 有 总 
创建 数据 表 使 用 的 是 __。 
临时 表 时 以 为 前 组 。 


、 选 择 题 


下 面 对 数 据 表 描述 正确 的 是 

A. 在 SQL Server 中 ， 一 个 数据 库 中 可 以 有 重 名 的 表 
B. 在 SQL Server 中 ， 一 个 数据 库 中 表 名 是 唯一 的 
C. 数据 表 通 常 都 以 数字 来 命名 

D. 以 上 都 不 对 

下 面 对 创 建 表 的 描述 正确 的 是 : 

A. 可 以 使 用 CREATE 语句 创建 不 带 字段 的 空 表 
B. 在 创建 表 时 ， 就 可 以 为 表 设 置 自动 增长 字段 
C. 在 创建 表 时 ， 字 段 名 可 以 重复 

D. 以 上 都 对 

下 面 对 修 改 数据 表 的 描述 正确 的 是 § 

A. 可 以 修改 表 中 字段 的 数据 类 型 

B. 可 以 删除 表 中 的 字段 

C. 可 以 将 表 重 命名 

D. 以 上 都 对 


| 


| 


三 、 问 答题 


简 述 SP_HELP 的 使 用 方法 。 
如 何 给 表 重 命名 。 
如 何在 其 他 文件 组 上 创建 数据 表 。 


、 操 作 题 


创建 名 为 test 的 数据 表 〈 自 定义 表 结 构 )， 并 对 表 做 如 下 操作 。 
(1) 将 表 中 的 第 1 个 字段 删除 。 

(2) 将 表 中 的 第 2 个 字段 改名 。 

(3) 将 表 test 的 名 字 更 改 成 newtest。 
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在 SQL Server 数据 库 中 ,约束 是 对 表 中 数据 制约 的 一 种 手段 。 通 过 约束 的 帮助 ， 可 以 
增强 表 中 数据 的 有 效 性 以 及 完整 性 。 约 束 可 以 理解 成 是 一 种 规则 或 者 要 求 ， 它 规定 了 在 数 
据 表 中 哪些 字段 可 以 输入 什么 样 的 值 。 在 现实 生活 中 , 也 有 很 多 与 约束 类 似 的 例子 ， 比 如 ， 
在 马路 上 分 为 机 动车 道 和 非 机 动车 道 ， 在 机 动车 道上 只 能 机 动车 通行 ， 在 非 机 动车 道上 只 


能 非 机 动车 通行 ， 否 则 就 会 造成 重大 的 交通 事故 。 
本 章 的 主要 知识 点 如 下 : 
口 为 什么 要 使 用 约束 
口 主键 约束 
口 外 键 约束 
口 默认 值 约束 
口 检查 约束 
口 唯一 约束 
口 非 空 约束 


4.1 为 什么 要 使 用 约束 


在 第 3 章 中 已 经 学 习 过 如 何在 SQL Server 数据 库 中 创建 数据 表 了 ， 显 然 没 有 约束 也 
是 可 以 创建 数据 表 的 。 那么 ， 为 什么 还 要 学 习 给 表 设 置 约束 呢 ? 约束 都 能 给 我 们 带 来 哪些 


好 处 呢 ? 带 着 这 两 个 问题 ， 我 们 进入 下 面 的 学 习 。 


首先 解决 第 1 个 问题 , 为 什么 要 使 用 约束 。 下 面 看 一 张 学 生 信息 表 中 的 数据 ， 如 表 4-1 


所 示 。 
表 4-1 学 生 信息 表 

学 号 性 别 专 业 
120001 男 软件 技术 
120001 男 软件 技术 
120002 女 会 计 
120003 23 工程 管理 
120004 男 土木 工程 
120005 男 


在 表 4-1 的 数据 中 找 找 问题 ， 看 看 是 否 存在 如 下 问题 呢 ? 
(1) 有 两 个 学 号 是 120001 的 学 生 记 录 ， 并 且 两 条 记录 完全 相同 。 
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(2) 学 号 是 120002 的 王 五 同学 的 年 龄 是 210。 
(3) 学 号 是 120003 的 刘 六 同学 的 性 别 是 23。 
(4) 学 号 是 120005 的 久久 同学 的 专业 信息 是 空 的 。 
上 面 的 4 个 问题 是 向 数据 表 中 录入 数据 时 常见 的 错 录 或 漏 录 的 问题 。 那 么 ， 如 何 避 免 
这 些 问 题 的 产生 呢 ? 这 就 是 我 们 要 解决 的 第 2 个 问题 了 ， 约 束 给 我 们 带 来 的 好 处 。 先 认识 
一 下 SQL Server 中 的 约束 ， 再 考虑 应 该 用 哪 种 约束 解决 上 面 的 4 个 问题 吧 。 
在 SQL Server 数据 库 中 主要 包括 主键 约束 、 外 键 约束 、 默 认 值 约束 、 检 查 约束 、 唯 一 
约束 和 非 空 约束 。 这 6 种 约束 的 给 我 们 带 来 的 好 处 如 下 。 
口 主键 约束 : 在 一 张 数 据 表 中 只 能 设置 一 个 主键 约束 。 它 主要 是 用 来 确保 列 的 唯一 
性 的 。 另外， 设置 主键 约束 的 列 不 能 够 为 空 。 主 键 约束 的 列 可 以 由 1 列 或 多 列 来 
组 成 ， 由 多 列 组 成 的 主键 被 称 为 联合 主键 。 有 了 主键 约束 ， 在 数据 表 中 就 不 用 担 
心 出 现 重复 的 行 了 。 
口 外 键 约束 : 是 这 6 种 约束 中 唯一 一 个 涉及 到 两 张 表 的 约束 。 它 可 以 确保 两 张 表 中 
数据 的 一 致 性 和 完整 性 。 比 如 : 在 录入 学 生 的 专业 信息 时 ， 只 能 录入 学 校 已 经 开 
设 的 专业 ， 不 能 随意 录入 专业 。 也 就 是 说 ， 有 了 外 键 约 束 ， 在 数据 表 中 就 可 以 保 
证 数据 的 准确 性 了 。 
口 默认 值 约束 : 一 张 表 中 可 以 设置 多 个 默认 值 约束 的 列 。 但 是 ， 每 一 个 列 默认 值 只 
有 一 个 。 这 就 好 像 是 家 里 的 电源 开关 ， 默 认 的 时 候 是 关 着 的 ， 使 用 的 时 候 是 开 着 
的 。 有 了 默认 值 约束 ， 在 数据 表 中 设置 默认 值 约束 的 列 就 不 会 出 现 空 值 了 。 
口 唯一 约束 : 一 张 表 中 可 以 设置 多 个 唯一 约束 的 列 。 唯 一 约束 与 主键 约束 有 些 相 似 ， 
也 是 确保 列 值 不 重复 的 ， 不 同 的 是 唯一 约束 可 以 允许 列 有 空 值 存在 并 且 可 以 设置 
多 个 唯一 约束 。 有 了 唯一 约束 ， 就 可 以 确保 数据 表 中 某 些 列 的 值 是 唯一 的 。 
口 检查 约束 : 一 张 表 中 也 可 以 设置 多 个 检查 约束 。 它 主要 是 用 来 规定 表 中 某 一 列 输 
入 值 的 取 值 范围 的 。 它 可 以 规定 在 某 一 列 只 能 输入 哪些 值 ， 或 者 一 个 具体 的 范围 。 
比如 : 在 学 生性 别 列 上 只 能 输入 “ 男 ” 或 “ 女 ” 在 学 生年 龄 列 上 只 能 输入 30 以 
下 的 数 。 有 了 检查 约束 ， 就 可 以 避免 向 表 中 输入 一 些 垃圾 数据 了 。 
口 非 空 约束 : 一 张 表 中 也 可 以 设置 多 个 非 空 约束 。 它 主要 是 用 来 规定 某 一 列 必须 要 
输入 值 。 有 了 非 空 约束 ， 就 可 以 避免 表 中 出 现 空 值 了 。 
知道 了 这 6 种 约束 的 好 处 了 ， 现 在 就 来 看 看 如 何 设置 约束 来 避免 出 现 表 4-1 中 出 现 的 
4 个 问题 了 。 
(1) 针对 学 号 120001 的 张 三 有 两 条 重复 的 记录 ， 可 以 将 学 号 列 设置 成 主键 约束 。 这 
样 ， 就 能 够 确保 学 号 在 学 生 信 息 表 中 是 唯一 的 并 且 又 没有 空 值 。 
(2) 针对 学 号 120002 的 王 五 年 龄 是 210 的 问题 ， 可 以 在 年 龄 列 上 设置 一 个 检查 约束 ， 
确保 年 龄 在 18 一 45 之 间 。 
(3) 针对 学 号 120003 的 刘 六 性 别 是 23 的 问题 ， 也 可 以 通过 设置 检查 约束 来 解决 。 将 
性 别 列 设置 成 只 能 输入 “ 男 ” 或 “ 女 ”。 
(4) 针对 学 号 120005 的 久久 专业 为 空 的 问题 ， 可 以 将 专业 列 设置 非 空 约束 。 这 样 ， 
就 必须 在 录入 学 生 信息 时 ， 录 入 专业 信息 了 。 另 外 ， 通 常情 况 下 ， 学 校 的 管理 系统 中 都 会 
有 一 张 专业 信息 表 ， 学 生 信息 表 中 的 专业 都 来 源 于 专业 信息 表 。 这 时 ， 就 可 以 将 学 生 信息 
表 中 的 专业 设置 成 与 专业 信息 表 中 专业 信息 的 外 键 约束 。 


。T1 。 
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看 来 ， 约 束 还 真 的 有 用 哟 ! 刚才 讲 的 都 是 纸上谈兵 ， 下 面 让 我 们 来 动手 在 SQL Server 
中 真正 地 使 用 这 些 约束 吧 。 


4.2 主键 约束 一 一 PRIMARY KEY 


使 用 主键 约束 请 记 住 PRIMARY KEY 这 个 关键 字 , 通常 主键 的 名 字 都 会 以 PK 开头 的 。 
主键 约束 几乎 在 每 张 数据 表 都 存在 ， 主 要 是 用 来 确保 表 中 数据 的 唯一 性 。 主 键 约束 在 使 用 
时 也 很 简单 ， 可 以 通过 SQL 语句 来 创建 也 可 以 通过 SQL Server 的 企业 管理 器 来 创建 。 下 
面 就 尽 你 所 能 来 模仿 吧 。 


4.2.1 在 建 表 时 直接 加 上 主键 约束 


主键 约束 在 创建 表 时 很 容易 就 可 以 加 上 ， 但 是 ， 要 记 住 主键 约束 在 每 张 数 据 表 中 只 有 
一 个 。 在 设置 主键 约束 时 要 先 确定 表 中 主键 约束 是 单列 的 主键 约束 还 是 由 多 列 组 成 的 联合 
主键 约束 。 在 创建 主键 约束 时 可 以 通过 下 面 两 种 语法 规则 来 创建 。 


1. 使 用 SQL 语句 设置 列 级 主键 约束 


所 谓 列 级 主键 约束 就 是 在 数据 列 的 后 面 直接 使 用 PRIMARY KEY 关键 字 设 置 , 并 不 指 
明 主键 约束 的 名 字 。 这 时 的 主键 约束 名 字 是 由 数据 库 系 统 自动 生成 的 。 具 体 的 语法 如 下 
所 示 。 

CREATE TABLE NAME table name 

COLUMN NAME1 DATATYPE PRIMARY KEY, 


COLUMN NAME2 DATATYPE, 
COLUMN NAME3 DATATYPE 


上 上 面 的 语法 规则 读者 可 以 看 出 来 ， 它 与 创建 表 时 的 语法 规则 是 类 似 的 ， 只 是 在 需要 设 
置 主键 约束 列 的 后 面 加 上 了 PRIMARY KEY 而 已 。 但是， 如 果 需 要 设置 联合 主键 时 就 不 能 
再 使 用 设置 列 级 主键 约束 的 方法 了 。 


2. 使 用 SQL 语句 设置 表 级 主键 约束 


所 谓 表 级 主键 约束 ， 也 是 在 创建 数据 表 时 创建 的 ， 但 是 可 以 指定 主键 约束 的 名 字 。 另 
外 ， 设 置 表 级 主键 约束 可 以 设置 联合 主键 。 具 体 的 语法 如 下 : 


CREATE TABLE NAME table name 

( 

COLUMN NAME1 DATATYPE, 

COLUMN NAME2 DATATYPE, 

COLUMN NAME3 DATATYPE 

[CONSTRAINT constraint name] PRIMARY KEY (COLUMN NAME1, COLUMN NAME2,...) 


这 里 ，[CONSTRAINT constraint name] 是 用 来 设置 主键 约束 名 字 的 ， 可 以 省 略 。 省 略 
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后 主键 约束 的 名 字 依然 是 由 系统 自动 产生 的 。 通 常 ， 在 创建 表 级 主键 约束 时 ， 都 会 为 其 指 
定 一 个 名 字 ， 这 样 能 够 方便 对 约束 的 管理 。 在 PRIMARY KEY 后 面 的 括号 里 可 以 放置 1 个 
或 多 个 用 于 设置 主键 约束 的 列 ， 这 些 列 之 间 用 逗号 隔 开 就 可 以 了 。 

上 面 的 两 个 语法 格式 记 住 了 吗 ? 现在 就 要 开始 演练 了 ! 

【示例 1】 创建 一 个 水 果 商 店 的 商品 信息 表 ， 表 结构 如 表 4-2 所 示 。 并 分 别 通过 设置 
列 级 和 表 级 主键 约束 的 方法 为 编号 设置 主键 约束 。 


表 4-2 水 果 信 息 表 (fruitinfo) 


编号 列 名 数据 类 型 中 文 释义 
| id INT 编号 
和 name VARCHAR(20) 名 称 
3 price DECIMAL(6,2) 价格 
4 origin VARCHAR(20) 产地 
5 tel VARCHAR(15) 供应 商 联系 方式 
6 remark VARCHAR(200) 备注 


(1) 使 用 列 级 约束 设置 的 方法 创建 主键 约束 ， 语 句 如 下 : 


CREATE TABLE fruitinfo 

( 
id INT PRIMARY KEY, 
name VARCHAR(20), 
Price DECIMAL(6,2), 
origin VARCHAR(20), 
tel VARCHAR (15), 
remark VARCHAR(200) 

) 


(2) 使 用 表 级 约束 设置 的 方法 创建 主键 约束 ， 语 句 如 下 : 


CREATE TABLE fruitinfo 

( 
id INT, 
name VARCHAR(20), 
Price DECIMAL(6,2), 
origin VARCHAR(20), 
tel VARCHAR (15), 
remark VARCHAR(200), 
PRIMARY KEY (id) 

) 


这 里 ，PRIMARY KEY(id) 就 代表 了 给 id 列 设置 了 主键 约束 ， 但 是 没有 给 主键 约束 起 
名 字 。 读 者 可 以 试 着 给 主键 约束 起 个 名 字 。 

通过 上 面 的 两 种 方式 就 在 数据 库 中 创建 了 水 果 信息 表 (fruitinfo)， 下 面 通过 sys.objects 
表 来 查看 一 下 是 否 真 的 创建 成 功 了 。 查 询 语句 如 下 所 示 ， 如 果 不 清楚 查询 语句 的 写法 可 以 
参考 第 3 章 中 关于 sys.objects 表 的 讲解 。 

select name,type,type desc 


from sys.objects 
where parent object id=object id('fruitinfo') and type='PK'; 


。73° 
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结果 如 图 4.1 所 示 。 
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图 4.1 查看 主键 是 否 创 建 界面 
在 图 4.1 所 示 的 结果 中 ， 可 以 看 到 name 列 中 显示 的 就 是 主键 的 名 字 。 
4.2.2 在 修改 表 时 加 上 主键 约束 


如 果 在 创建 数据 表 时 忘记 了 设置 主键 约束 或 者 是 还 没有 想 好 将 哪个 列 设置 成 主键 约 
束 ， 不 用 怕 ， 主 键 约束 也 可 以 在 修改 表 的 时 候 加 上 。 但 是 ， 在 修改 表 时 添加 主键 约束 ， 就 
必须 要 使 用 表 级 约束 的 方式 来 添加 了 。 

修改 表 时 添加 主键 约束 ， 依 然 使 用 ALTER TABLE 语句 来 完成 。 具 体 语法 如 下 : 

ALTER TABLE table name 

ADD CONSTRAINT pk name PRIMARY KEY(column namel, column name2,...) 

其 中 : 

口 pk_name: 是 要 设置 的 主键 名 称 。 

口 column_namel, column_name2,...: 是 要 设置 成 主键 的 列 ， 可 以 是 1 到 多 个 列 ， 多 

个 列 之 间 用 去 号 隔 开 。 

有 了 语法 ， 就 让 我 们 来 练习 一 下 吧 。 

【示例 2】 将 水 果 信息 表 (fruitinfo) 的 编号 列 设置 成 主键 。 假 设 水 果 信息 表 已 经 存在 
但 是 没有 设置 主键 。 

使 用 ALTER TABLE 语句 设置 主键 ， 语 名 如下: 


ALTER TABLE fruitinfo 
ADD CONSTRAINT pk fruitinfo PRIMARY KEY (id); -- 添 加 主键 约束 


上 面 的 语句 执行 成 功 后 ,就 可 以 为 fruitinfo 表 创建 一 个 名 为 pk_fruitinfo 的 主键 约束 了 。 
通过 sys.objects 表 查 看 创建 后 的 结果 如 图 4.2 所 示 。 查 询 语句 与 示例 1 相同 。 

通过 图 4.2 的 所 示 的 结果 ,就 可 以 看 到 刚才 创建 的 pk_fruitinfo 主键 约束 已 经 加 入 到 水 
果 信息 表 中 了 。 


4.2.3 ”去除 主键 约束 
如 果 表 中 的 主键 约束 出 现 误 加 的 情况 或 者 想 换 一 个 列 作为 主键 约束 ， 首 先 要 做 的 是 去 


。T4 。 
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图 4.2 添加 主键 约束 pk_fruitinfo 


除 表 中 当前 的 主键 约束 ， 然 后 再 重新 创建 主键 约束 。 将 表 中 的 主键 约束 去 除 是 很 容易 的 事 
情 ， 就 像 买 东西 的 时 候 需 要 挑 来 挑 去 ， 扔 东西 的 时 候 只 需要 直接 扔 到 垃圾 箱 就 可 以 了 。 去 
除 主键 约束 的 语法 如 下 : 

ALTER TABLE table name 

DROP CONSTRAINT pk_name 

其 中 

口 table name: 要 去 除 主键 约束 的 表 名 。 

口 pk_name: 主键 约束 的 名 字 。 


全 注意 : 如 果 想 去 除 某 张 表 中 的 主键 约束 时 ,不 知道 主键 约束 的 名 字 , 可 以 通过 sys.objects 
先 查看 一 下 表 中 主键 约束 的 名 字 ， 然 后 再 删除 。 


【示例 3】 将 水 果 信息 表 (fruitinfo》 中 的 主键 约束 去 除 。 

在 示例 2 中 为 水 果 信 息 表 添加 的 主键 名 称 是 pk_fruitinfp， 有 了 主键 名 称 ， 去 除 主键 就 
很 容易 了 。 语 句 如 下 : 

ALTER TABLE FRUITINFO 

DROP CONSTRAINT PK FRUITINFO; 

执行 上 面 的 语 名 后， 水 果 信 息 表 (fruitinfo》 中 就 没有 主键 约束 了 。 
4.2.4 使 用 企业 管理 器 轻松 使 用 主键 约束 


前 面 的 几 个 小 节 里 , 读者 已 经 知道 了 如 何 使 用 SQL 语句 来 添加 和 去 除 主键 约束 。 可 能 
读者 觉得 用 语句 操作 主键 约束 太 繁琐 了 ， 还 要 记 那 么 多 的 语法 。 现 在 就 教 给 大 家 一 个 最 简 
单 的 方法 ， 在 企业 管理 器 中 使 用 主键 约束 。 为 了 能 够 显示 出 企业 管理 器 的 方便 性 ， 用 企业 
管理 器 重新 完成 示例 1 一 示例 3 的 练习 。 

【示例 4】 使 用 企业 管理 器 在 创建 水 果 信息 表 时 添加 主键 约束 。 

使 用 企业 管理 器 完成 主键 约束 的 添加 ， 只 需 如 下 两 个 步骤 即 可 。 

(1) 创建 水 果 信 息 表 

打开 企业 管理 器 ， 展 开 chapter4 数据 库 ， 然 后 右 击 表 文 件 夹 ， 在 弹出 的 右键 菜单 中 选 
择 “ 新 建 表 ” 选 项 ， 界 面 如 图 4.3 所 示 。 
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文件 四 编辑 下 ) 视图 WD， 项目) 调试 @@) 表 设计 器 C) 工具 名 窗口 思 社区 CC) 帮助 0 
BSE EE 
回回 所 


TIDY-952TAE_ - -dbo- Table_1 vx 
列 名 数据 类 型 区 许 Nul 值 
上 Cc 


ver Nanagenent Studie 


图 4.3 ”新 建 表 窗口 


在 图 4.3 所 示 的 界面 中 ， 录 入 表 4-2 所 示 表 结构 中 的 字段 信息 。 录 入 后 效果 如 图 4.4 
所 示 。 


icrosoft SQL Server Nanagement Studio =Io|x| 
文件 到 ) 编辑 到 ) 视图 WD 项目) 调试 @@) 表 设 计 器 LD) 工具 CD) 窗口 @) 社区 C) 
帮助 0 


也 新建 查询 中 | 山 | 轧 也 加 所 | 切 台 与 | 过 百 


田 国 系统 数据 库 
田 国 数据 库 快照 
田 辐 KeportserverSas: 


numeric(6, 2) 
origin varchar(20) 


图 国 RepertserverShs: 
varchar(15) 


chapter4 
田 国 数据 库 关系 图 


图 4.4 水 果 信息 表 的 结构 


(2) 添加 主键 约束 

有 了 图 4.4 的 表 结 构 ， 就 意味 着 已 经 前 进 一 大 步 了 。 下 面 再 向 前 冲刺 一 下 就 完成 了 。 
在 图 4.4 所 示 的 界面 中 ， 右 击 要 设置 成 主键 约束 的 列 ， 如 图 4.5 所 示 。 

在 图 4.5 所 示 的 右键 菜单 中 ， 选 择 “ 设 置 主键 ”选项 ， 即 可 完成 主键 的 设置 。 如 果 是 
多 列 需要 设置 成 联合 主键 ， 可 以 按 下 Ctrl 键 加 选 要 设置 成 主键 的 列 ， 然 后 再 使 用 右键 菜单 
中 的 “设置 主键 选项 设置 。 另 外 , 也 可 以 选中 要 设置 主键 的 列 后 , 单 击 工具 栏 上 的 图 标 
来 设置 主键 。 设 置 成 主键 的 列 ， 在 列 的 前 面 会 出 现 一 个 小 钥匙 的 图 标 。 这 样 就 完成 了 主键 
的 设置 。 还 有 ， 最 后 一 定 要 保存 数据 表 ! 
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了 microseft SQL Server Nanagenment Studio =Iolx| 
文件 时 ”编辑 到) 视图 WD” 项目) 调式) 表 设 计 器 L) 工具 窗口 思 社区 加 ) 
帮助 0 设置 主键 


到 芒 回 各 | 剖 5 
EF 
IDr-9527\n. .. dbo. Table_1* 
> 列 名 
name 
田 加 数据 库 快照 
国 罩 kepertserverSas: 
田 国 keportserverShs: 
日 国 chapter4 tol 
田 国 数据 库 关系 图 remark 司 索引 / 键 四 ) 
日 国 表 [| 2 让 
田 国 系统 表 列 属 性 | I 
dbo. Eruit = 司 | xm 索引 Q@)... 
四 国名 El 国 omcr 约束 
回回 可 篇 和 性 ESD | aro 
加 电 生成 更 改 购 本 G)…. 


图 4.5 列 操作 的 右键 菜单 


【示例 S】 使 用 企业 管理 器 在 修改 水 果 信 息 表 时 添加 主键 约束 。 

在 修改 表 时 添加 主键 约束 就 更 加 简单 了 , 不 需要 创建 表 , 只 需要 打开 表 就 可 以 创建 了 。 
记 住 ， 还 是 两 个 步骤 。 

(1) 打开 要 添加 主键 约束 的 数据 表 的 设计 页 面 

打开 企业 管理 器 ， 展 开 chapter4 数据 库 ， 右 击 fruitinfo 表 ， 在 出 现 的 右键 菜单 中 选择 
“设计 ”选项 ， 打 开 表 的 设计 界面 ， 如 图 4.6 所 示 。 


Cement Studio 


st 
文件 中 编辑 下 ) 视图 Y) 项 目 人 E) 调式 W) 表 设 计 器 L) 工具 0) 窗口 WD 社区 C) 


帮助 0 
六 新 村 光 D | 启 孔 也 马上 包 | 忆 台电 | 怠 天 


1 [中间 记忆 回回 二 
TDY-952TVE .0. frwitinfes| sx 
EE 3 


a 
name varchar(20) 区 
oecmak6, 区 
origin Yarchar(20) [a 
tel 5 
emarl Ep 

首 


varchar(15) 
varchar(200) 


图 4.6 ”fruitinfo 的 设计 页 面 


(2) 设置 主键 约束 

在 图 4.6 所 示 的 界面 中 ， 右 击 要 设置 主键 的 id 列 ， 在 弹出 的 右键 菜单 中 选择 “设置 主 
键 ”选项 ， 即 可 完成 主键 约束 的 设置 。 设 置 后 的 效果 如 图 4.7 所 示 。 

【示例 6】 删除 水 果 信 息 表 中 的 主键 约束 。 

通过 前 面 的 示例 5 和 示例 6， 相 信 读 者 已 经 了 解 了 企业 管理 器 是 如 何 设置 主键 约束 的 
了 。 那 么 去 除 主键 约束 的 方法 ， 你 想到 了 吗 ? 很 简单 ， 就 是 右 击 设置 为 主键 约束 的 列 ， 在 


。 TIT 。 


第 2 篇 表 操 作 基础 


so: Si Server Nanagesent Studio 
i 编 可 到 ) 视图 WW) 项 目 E) 调试 中) 表 设 计 器 LD) 工具 CD) 窗口 ) 社区 人 C) 
帮助 0D 
了 四 DDBISaSlas 
de 


name varchar(20) a 

田 国 数据 库 快 照 price decimak6, 2) 加 
田 ReportServersls: i 2 加 
tel a 

remark Ea 

三 


图 4.7 设置 主键 约束 后 的 效果 
弹出 的 右键 菜单 中 (如 图 4.8 所 示 )， 选 择 “删除 主键 ”选项 ， 即 可 完成 删除 主键 的 操作 。 


crosoft SQL Server Nanagesent Studio =|DIxl 


文件 EE) 编辑 人 E) 视图 WD 项目) 调式 0) 表 设 计 器 LL) 工具 CD) 窗口 岂 ) 社区 C) 
才 助 00) 


;也 新 建 查询 四 | 让 | 思 遍 加 | 记 | 区 加 局 | 怠 瑟 
;昌国 冯 司 网 到 回 加 呈 


辐 omcx 约束 加 
S 昌 索引 @) | 
二 生成 玉 veG). 上 习 


3 


图 4.8 列 操作 的 右键 菜单 


删除 主键 约束 后 ， 原 来 的 主键 约束 列 前 面 的 小 钥匙 就 消失 了 。 不 要 高 兴 得 太 早 哟 ， 
得 要 保存 对 表 的 操作 才 可 以 啊 ! 


4.3 ”外 键 约束 一 一 FOREIGN KEY 
外 键 约束 是 唯一 一 个 与 两 张 表 相关 的 约束 ， 它 主要 的 用 途 就 是 制约 数据 表 中 的 数据 ， 
能 够 确保 数据 表 中 数据 的 有 效 性 。 这 就 好 像 是 去 书店 购买 图 书 , 只 能 购买 书店 中 有 的 图 书 ， 
不 能 想 买 什么 书 就 有 什么 书 。 回 到 数据 库 中 也 是 一 样 ， 在 购买 图 书 时 只 能 选择 书店 中 显示 
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的 图 书 列表 中 的 图 书 ， 无 法 选择 其 他 的 图 书 。 如 果 对 这 样 的 数据 不 加 以 约束 ， 就 会 出 现 一 
些 购 买书 店 中 没有 的 图 书 这 样 的 无 效 数据 。 


4. 


大 | 


a 


3.1 在 建 表 时 直接 加 上 外 键 约束 

外 键 约 束 相对 于 其 他 的 约束 ， 在 设置 时 有 些 复杂 ， 但 是 它 又 是 一 个 比较 重要 的 约束 。 
此 ， 请 读者 要 认真 学 习 外 键 约束 的 设置 方法 。 外 键 约束 在 创建 表 时 就 可 以 添加 ， 但 有 一 
前 提 就 是 与 这 个 外 键 约束 相关 的 那 张 数 据 表 也 已 经 存在 了 。 否 则 ， 就 会 出 现 错误 了 ! 在 


创建 表 时 加 上 外 键 约束 的 语法 如 下 : 


需要 读者 记忆 的 。 既 然 如 此 ， 就 


例 


CREATE TABLE table name 

( 

col namel datatype, 

col name2 datatype, 

col name3 datatype, 

CONSTRAINT fk name FOREIGN KEY (col namel,..) REFERENCES 
referenced table name (ref col namel,...) 


) 

其 中 ;: 

口 人 k_name: 外 键 约束 的 名 字 ， 通 常 以 代 开头 。 

口 col_namel: 要 设置 成 外 键 约束 的 列 名 ， 可 以 由 多 个 列 组 成 。 

口 referenced table name: 被 引用 的 表 名 。 

口 ref col namel: 被 引用 的 表 中 的 列 名 ， 也 可 以 由 多 个 列 组 成 。 

从 上 面 的 语法 中 ， 可 以 看 出 语法 中 的 前 半 部 分 都 是 创建 表 时 用 到 的 ， 只 有 最 后 一 句 是 
-下 吧 。 

在 演练 之 前 ， 先 来 创建 一 下 演练 时 要 用 的 数据 表 。 仍 然 用 表 4-2 所 示 的 水 果 信 息 表 为 

， 再 为 其 创建 一 张 水 果 供 应 商 信 息 表 ， 并 将 其 水 果 信息 表 中 的 供应 商 联 系 方式 换 成 水 果 


供应 商 信息 表 中 的 编号 。 这 样 ， 两 张 表 的 结构 就 变 成 了 表 4-3 和 表 4-4 的 样子 。 


表 4-3 ”水果 供应 商 信息 表 (supplierinfo) 


编号 数据 类 型 中 文 释义 
1 INT 编号 
2 VARCHAR(20) 供应 商 名 称 
3 VARCHAR(15) 电话 
4 VARCHAR(200) 备注 
表 4-4 水 果 信 息 表 (fruitinfo) 
编号 列 名 数据 类 型 中 文 释义 
1 id INT 编号 
2 name VARCHAR(20) 名 称 
3 price DECIMAL(6,2) 价格 
4 origin VARCHAR(20) 产地 
5 supplierid INT 供应 商 编 号 
6 remark VARCHAR(200) 备注 
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创建 水 果 供应 商 信 息 表 的 语句 如 下 : 


CREATE TABLE supplierinfo 
( 
id int primary key, 
name Varchar (20), 
tel Varchar (15), 
remark varchar (200) 


将 上 面 的 语句 在 SQL Server 的 企业 管理 器 中 执行 一 下 , 即 可 完成 水 果 供 应 商 信息 表 的 
创建 。 示 例 7 将 演示 为 表 4-4 中 的 水 果 信 息 表 创建 外 键 约束 。 

【示例 7】 在 创建 水 果 信 息 表 (fruitinfo) 时 ， 为 供应 商 编号 创建 外 键 约束 。 

根据 表 4-4 的 结构 ， 创 建 水 果 信 息 表 的 语句 如 下 : 


CREATE TABLE fruitinfo 


( 
id INT PRIMARY KEY, 
name VARCHAR(20), 
price DECIMAL(6,2), 
origin VARCHAR(20), 
supplierid INT, 
remark VARCHAR(200) 
CONSTRAINT fk fruit FOREIGN KEY (supplierid) REFERENCES supplierinfo(id) 
-创建 外 键 约束 
) 


执行 上 面 的 语句 后 ， 就 可 以 为 水 果 信息 表 〈fruitinfo〉 中 的 供应 商 编号 创建 外 键 约束 。 
查看 是 否 创建 了 外 键 约束 ， 依 然 可 以 使 用 sys.objects 这 个 系统 表 来 查看 。 只 是 查询 语句 有 
些 变 化 ， 语 句 如 下 : 


select name,type,type desc 
from sys.objects 
where parent object id=object id('fruitinfo') and type="'F'; 


-查找 在 fruitinfo 表 中 约束 类 型 是 外 键 约束 的 信息 
运行 结果 如 图 4.9 所 示 。 


Microsoft SQL Server Nanagenent Studie =|D| xj 
文件 全 编辑 下) 视图 WD 查询 @) 项 目 E) 调试 @) 工具 CD) 窗口 WD) 社区 人 


帮助 0 


也 币 建 查询 | 出 他 渤 串 | 让 | 芒 回 对 | 加 5 

Wy a | chepter4 ~| ?执行 DO 有 红 可 国 | 玉 怠 | 的 名 

SQLQueryl. s... ator (55))*| WIDN-SS27W. .0 fruitingo | vx 
select name,type, type_desc 

from 

where parent object 


_id=object_idl('fruitinfo') and type='F'; 


回 结果 | 已 消息 | 
|__| name |[ gpe | bpe desc 


| k_mu， WY FoREIGN_kEY_CONSTRAINT 


@ | YIn-es27 nssQLSERYER2008 (.. | WIDW-9527\Adninistrato. .. | chapter4 |00:00:00 |1 行 
就 绪 第 1 行 第 2 列 A 


图 4.9 查看 外 键 约束 
从 图 4.9 所 示 的 结果 可 以 看 出 ， 在 最 后 的 约束 信息 框 中 外 键 约束 全 fruit 已 经 创建 了 。 
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名 说明: 除了 通过 sys.objects 系统 表 来 查看 外 键 约束 之 外 ,还 可 以 通过 系统 表 sys.foreign_ 
keys 来 查看 。 但 是 ， 通 过 系统 表 sys.foreign keys 查看 出 来 的 外 键 约束 是 该 数据 
库 中 存在 的 所 有 外 键 约束 。 如 果 读 者 按照 示例 7 的 要 求 创建 了 外 键 约束 ， 在 当前 
使 用 的 数据 库 中 也 没有 其 他 的 外 键 约束 , 那么 , 查询 结果 就 和 图 4.10 所 示 的 结果 
= 


Ricrosoft SQL Server Nanagenent Studio 
文件 四 ”编辑 下 ) 视图 @) 查询 @) 项 目 EE) 调式 Wm) 工具 YI) 窗口 @ 社区 C) 帮助 0D 
卫 新建 查询 思 | 让 | 声 也 下 包 | 启 加 号 | 到 着 
于 | chapter4 -| 执行 WwW 》 a x 弹 旺 国 | 3 盐 | 蛤 贺 克 | 三 全 | 地 地 | 筷 遇 
‘SQLQueryl. s... ator (51))*| vx 
上 日 select * from sys.fore ey 
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3 
器 结果 | 过 消息 | 

name_| obeetid [pincipalid | :chema id [parent obiectid | ype | wpedese [ceasledae 
有 | 213575799 ”NULL 1 149575571 F FOREIGN_KEY_CONSTRAINT ”2012.10.01 09:31:20.607 


| WIDW-9527 \SSQLSERVER2008 (.. | WIDW-9527\Adninistrato. .. | chapter4 |00:00:00 |1 行 
行 1 列 31 


Ch 31 


图 4.10 ”从 sys.foreign_keys 中 查看 外 键 约束 


4.3.2 在 修改 表 时 加 上 外 键 约束 


如 果 已 经 创建 了 数据 表 ， 但 是 忘记 了 添加 外 键 约束 怎么 办 呢 ? 当然 是 不 需要 重新 创建 
数据 表 了 ， 只 需要 在 修改 表 的 语句 中 加 上 外 键 约束 就 可 以 了 。 在 修改 表 时 添加 外 键 约束 的 
语法 如 下 : 

ALTER TABLE table name 

ADD CONSTRAINT fk name FOREIGN KEY(col namel,..) REFERENCES 

referenced table name(ref col namel,...); 

其 中 : 

口 fk_name: 外 键 约束 的 名 字 ， 通 常 以 信 开头 。 

口 col_namel: 要 设置 成 外 键 约束 的 列 名 ， 可 以 由 多 个 列 组 成 。 

口 referenced table name: 被 引用 的 表 名 。 

口 ref_col namel: 被 引用 的 表 中 的 列 名 ， 也 可 以 由 多 个 列 组 成 。 


全 说 明 : 在 添加 外 键 约束 前 ， 要 确保 表 中 要 设置 外 键 约束 列 的 值 全 部 都 符合 引用 表 中 对 应 
的 列 值 ， 否 则 就 会 出 现 添加 外 键 约束 失败 的 错误 。 通常 情况 下 ,会 在 表 中 还 没有 
添加 数据 的 情况 下 ， 为 数据 表 添 加 约束 。 


下 面 就 使 用 上 面 的 语句 ， 来 演练 一 下 如 何在 已 经 存在 的 数据 表 上 添加 外 键 约束 吧 ! 

【示例 8】 假设 表 4-3、 表 4-4 中 所 示 的 水 果 供 应 商 信息 表 (supplierinfo〉 和 水 果 信 息 
表 (fruitinfo) 已 经 存在 ， 现 为 水 果 信 息 表 中 水 果 供 应 商 编号 〈suppelierid) 设置 成 与 水 果 
供应 商 编号 〈id) 的 外 键 。 

假设 水 果 信息 表 中 还 没有 数据 ， 这 样 在 设置 外 键 约束 就 不 会 出 现 错误 了 。 上 有 具体 的 设置 
语句 如 下 : 
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ALTER TABLE fruitinfo 
ADD CONSTRAINT fk supperlierid FOREIGN KEY (suppelierid) REFERENCES 
supplierinfo (id); 


执行 上 面 的 语句 后 ， 就 可 以 为 水 果 信息 表 fruitinfo 添加 外 键 约束 了 。 效 果 如 图 4.11 
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日 ALTER TABLE fruitinfo 
E22 CONSTRAINT fk supperlierid FOREIGN KEY(supplierid) REFERENCES supplierinfo(id); 


1 上 
轧 消息 | 

命令 已 成 功 完成 。 可 
yl 入 
回 查询 已 成 功 执行 。 | WIDW-9527\MSSQLSERVER2008 (.. | YITDY-9527\Adninistrate .| chapter4 |00:00:00 |0 行 


就 绪 行 3 列 1 Ch1 Ins /4 


图 4.11 添加 外 键 约束 化 supperlierid 的 效果 


4.3.3 去 除外 键 约束 


外 键 约束 既然 能 够 添加 ， 当 然 也 能 够 去 除 。 使 用 SQL 语句 去 除外 键 约束 时 ,最 重要 的 
就 是 要 知道 你 要 去 除 的 外 键 约束 名 称 。 这 样 好 像 是 在 商场 买 东西 ， 去 退货 时 都 需要 购买 凭 
证 一 样 。 但 是 ， 如 果真 的 记 不 起 来 了 ， 也 是 可 以 通过 会 员 卡 号 来 查找 的 。 对 于 数据 库 来 说 ， 
虽然 忘记 了 外 键 约束 的 名 字 ， 但 是 应 该 知道 要 去 除外 键 约束 的 表 名 。 在 前 面 的 内 容 中 已 经 
学 习 了 sys.objects 可 以 查看 外 键 约束 是 否 创建 成 功 。 当 然 ， 咱 也 可 以 通过 它 来 查看 外 键 约 
束 的 名 称 。 

如 果 已 经 知道 了 要 删除 的 外 键 约束 名 称 ， 那 么 ， 恭 喜 你 ， 你 已 经 完成 了 删除 外 键 约束 
一 般 的 工作 。 具 体 的 去 除外 键 约束 的 语句 如 下 : 

ALTER TABLE table name 

DROP CONSTRAINT fk name; 

其 中 : 

口 table name: 要 去 除外 键 约束 的 表 名 。 

口 全 name: 外 键 约束 的 名 字 。 

如 果 读 者 是 按照 顺序 来 阅读 本 章 内 容 的 ， 相 信 你 已 经 发 现 上 面 的 语法 与 去 除 主键 的 语 
法 是 相似 的 ， 只 不 过 在 约束 名 字 的 位 置 是 外 键 约 束 名 字 而 已 。 所 以 ， 只 要 掌握 了 去 除 主键 
约束 的 语法 ， 去 除外 键 约束 的 语法 就 不 需要 再 重复 记忆 了 。 

【示例 9】 去 除 水 果 信 息 表 (fruitinfo) 中 名 为 信 supperlierid 的 外 键 约束 。 

知道 了 外 键 约束 的 名 字 又 知道 外 键 约束 在 哪个 表 中 ， 只 需要 套用 一 下 去 除外 键 约束 的 
语法 就 可 以 去 除 约束 了 。 具 体 的 语句 如 下 : 

ALTER TABLE fruitinfo 

DROP CONSTRAINT fk supperlierid; 一 -删除 外 键 约束 
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通过 上 面 的 语句 ， 就 可 以 去 除外 键 约束 八 supperlierid。 运 行 效果 如 图 4.12 所 示 。 读 
者 还 可 以 通过 sys.objects 表 查 看 是 否 还 存在 该 外 键 约束 。 


S =l9|x| 
文件 时 ” 蝙 辑 隐 ) 视图 人 oO) 查询 @) 项 目 @) 调式 0) 工具 GD) 窗口 和 ) 社区 C) 
帮助 00 
;也 间 奸 查询 | 出 | 轧 逊 吕 | 记 | 区 加 本 | 乳牛 
; ?2 | chapter4 EE 
“SQLQueryl. s... ator (52))#| vx 
日 iLTER TABLE fruitinfo DROP CONSTRAINT fk supperlierid; 引 
EE 了 
«| 1 » 
六 消息 
命令 已 成 功 完成 。 
可 » 
@| WIDi-9s27 SsQLSERVER2008 (.. | WIDN-g527\Adninistrato. .. | chspter4 |00:00:00 |0 行 


就 绪 行 1 列 23 Ch 23 和 | 
图 4.12 ”删除 外 键 约束 全 supperlierid 的 效果 


4.3.4 使 用 企业 管理 器 轻松 使 用 外 键 约束 


通过 前 面 对 外 键 约束 的 讲解 ， 相 信 读 者 已 经 发 现 使 用 外 键 约束 比 主键 约束 还 麻烦 。 既 
然 主键 约束 可 以 使 用 企业 管理 器 来 操作 ， 那 么 ， 外 键 约束 也 是 可 以 通过 企业 管理 器 来 操作 
的 。 但 是 ， 即 使 可 以 用 企业 管理 器 来 操作 外 键 约束 ， 也 会 比 设置 主键 约束 稍微 复杂 一 些 。 
因此 , 下 面 的 内 容 请 读者 一 定 要 认真 地 学 习 哟 ! 为 了 对 比 企业 管理 器 操作 外 键 约束 的 方法 ， 
仍然 以 示例 7 一 示例 9 为 例 ， 重 新 使 用 企业 管理 来 管理 外 键 约束 。 由 于 在 企业 管理 器 中 ， 
在 创建 表 时 添加 外 键 约束 与 修改 表 时 添加 外 键 约束 都 非常 相似 ， 这 里 就 将 添加 和 修改 表 时 
添加 外 键 约束 都 合并 到 了 一 起 来 讲解 。 

【示例 10】 使 用 企业 管理 器 来 添加 水 果 信 息 表 (fruitinfo) 与 水 果 供应 商 信息 表 
(supplierinfo〉 的 外 键 约束 。 

在 企业 管理 器 中 ， 完 成 添加 或 修改 水 果 信 息 表 添加 外 键 约束 需要 如 下 3 个 步 又 。 

(1) 打开 水 果 信 息 表 的 设计 页 面 

在 企业 管理 器 中 ， 使 用 鼠标 右 击 水 果 信息 表 , 在 弹出 的 右键 菜单 中 选择 “设计 ”选项 ， 
出 现 水 果 信 息 表 的 设计 页 面 ， 如 图 4.13 所 示 。 如 
果 是 添加 表 ， 要 先 在 企业 管理 器 中 的 数据 库 界 面 
中 右 击 表 节 点 ， 在 弹出 的 右键 菜单 中 选择 “新 建 
表 ” 选 项 ， 创 建 后 的 效果 也 与 图 4.13 一 致 。 

(2) 打开 表 的 关系 页 面 

在 图 4.13 所 示 的 页 面 中 , 使 用 鼠标 右 击 表 设 
计 页 面 ， 在 弹出 的 右键 菜单 中 选择 “关系 ”选项 ， 
出 现 图 4.14 所 示 的 页 面 。 

(3) 添加 外 键 约束 

如 果 操 作 到 了 图 4.14 所 示 的 界面 ， 那 么 ， 现 在 就 可 以 添加 外 键 约束 。 在 图 4.14 所 示 
的 界面 中 ， 单 击 “ 添 加 ”按钮 ， 效 果 如 图 4.15 所 示 。 

现在 可 以 回忆 一 下 ， 外 键 约束 是 创建 两 张 表 中 的 字段 关系 的 。 那 么 ， 如 何 设置 呢 ? 很 简 
单 ， 在 图 4.15 所 示 的 界面 中 ， 单 击 “ 表 和 列 规范 ”后 面 的 按钮 ， 弹 出 如 图 4.16 所 示 的 界面 。 


图 4.13 水 果 信 息 表 (fruitinfo) 的 设计 页 面 
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日 ( 至 规 ) 
日 医 现 阳 

在 外 了 娃 或 重新 启用 时 检查 二 是 
日 标识 

TRfraitingo fraitinge 

日 表 设 计 各 
回 NSERT 和 WPDATE 规范 

强制 外 键 的 束 是 

强制 用 于 复制 是 


添加 (A) MB ) 
图 4.15 添加 外 键 关系 界面 
从 图 4.16 所 示 的 界面 中 ， 可 以 看 到 在 图 的 左 侧 是 主键 表 ， 右 侧 是 外 键 表 。 根 据 本 题 的 
要 求 给 水 果 信 息 表 添加 外 键 约 束 ， 那 么 ， 外 键 表 就 是 水 果 信 息 表 (fruitinfo)， 主 键 表 就 是 
水 果 供 应 商 信息 表 (supplierinfo)。 根 据 要 求 在 图 4.16 中 填 入 必要 信息 后 ， 效 果 如 图 4.17 
所 示 。 


EE 3 
关系 名 中: 


[Fk_fruitino_supplierinfo| 


图 4.16 表 和 列 的 设置 图 4.17 添加 表 和 列 的 信息 后 的 效果 
在 图 4.17 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 完成 对 水 果 信息 表 外 键 约束 的 设置 。 此 外 ， 还 
可 以 在 图 4.17 中 的 关系 名 处 ， 重 新 定义 外 键 约束 的 名 字 。 
这 样 ， 通 过 上 面 讲解 的 3 个 步骤 就 可 以 完成 对 外 键 约束 的 设置 。 读 者 可 以 发 现在 企业 
管理 器 中 设置 外 键 约束 ， 就 比 使 用 创建 外 键 约束 的 语法 容易 得 多 了 ! 
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【示例 11】 使 用 企业 管理 器 删除 水 果 信 息 表 中 添加 的 外 键 约束 。 

在 企业 管理 器 中 , 删除 外 键 约束 要 比 添加 外 键 约 束 容易 得 多 。 主要 分 为 如 下 3 个 步骤 。 

(1) 打开 水 果 信息 表 的 设计 页 面 

水 果 信 息 表 的 设计 页 面 ， 读 者 已 经 不 陌生 了 吧 ， 在 示例 10 中 就 已 经 见 过 了 ， 这 里 不 
再 多 说 。 按 照 示例 10 的 操作 打开 水 果 信 息 表 的 设计 页 面 就 可 以 了 。 

(2) 打开 关系 界面 

水 果 信 息 表 的 关系 界面 也 不 陌生 吧 ? 现在 把 它 打开 吧 。 效 果 如 图 4.18 所 示 。 


图 4.18 水 果 信 息 表 的 关系 界面 


(3) 删除 外 键 约束 

从 图 4.18 所 示 的 界面 中 ， 可 以 看 出 在 水 果 信 息 表 中 只 有 一 个 外 键 约束 ， 就 是 在 示例 
10 中 创建 的 。 现 在 要 将 其 删除 ， 只 需要 选中 这 个 外 键 约束 的 名 字 ， 然 后 单 击 “ 删 除 ” 按 钮 
即 可 将 外 键 约束 去 除了 。 


4.4 ”默认 值 约束 一 一 DEFAULT 
如 果 不 学 习 数 据 库 ， 默 认 值 约 束 的 这 个 称呼 是 不 会 听 过 的 。 但 是 ， 默 认 值 应 该 是 听 说 


过 的 吧 ? 当 在 网 上 注册 用 户 时 , 总 会 有 一 些 信 息 是 你 还 没有 填写 就 已 经 显示 出 来 的 , 比如 : 
性 别 ， 通 常会 默认 一 个 “ 男 ” 或 “ 女 ” 是 否 同意 使 用 协议 ， 通 过 会 默认 一 个 “是 ” 那么 ， 


这 些 默 认 值 会 当 你 不 修改 这 些 信息 时 ， 直 接 填 入 到 你 的 注册 信息 中 的 。 当 去 旅游 景点 游玩 
时 ， 购 买 门票 如 果 不 出 示 一 些 特殊 的 优惠 证 件 ， 一 定 会 按照 旅游 景点 门票 的 实际 价格 付费 


的 。 那 么 ， 这 个 实际 价格 就 是 默认 值 。 把 这 些 默 认 值 放 到 数据 库 中 ， 就 被 称 为 了 默认 值 
约束 。 


4.4.1 在 建 表 时 添加 默认 值 约束 


默认 值 约束 在 创建 表 时 就 可 以 添加 ， 通 常 是 在 什么 情况 下 需要 添加 默认 值 约束 呢 ? 一 
般 添加 默认 值 约束 的 字段 有 两 种 比较 常见 的 情况 ， 一 种 是 该 字段 不 能 为 室 ， 一 种 是 该 字段 
添加 的 值 总 是 某 一 个 固定 值 。 比 如 : 当 用 户 注册 信息 时 ， 数 据 库 中 会 有 一 个 字段 来 存放 用 
户 的 注册 时 间 ， 注 册 时 间 实 际 上 就 是 当前 的 时 间 ， 因 此 ， 可 以 为 该 字段 设置 一 个 当前 时 间 
(可 以 通过 getdate() 函 数 来 设置 ) 为 默认 值 。 在 创建 表 时 添加 默认 值 约束 的 语法 如 下 : 
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CREATE TABLE NAME table _ name 

( 

COLUMN NAME]1 DATATYPE DEFAULT constant expression, 
COLUMN NAME2 DATATYPE, 

COLUMN NAME3 DATATYPE 


其 中 : 

口 DEFAULT: 默认 值 约束 的 关键 字 。 它 通常 放 在 字段 的 数据 类 型 之 后 。 

口 constant_expression: 常量 表达 式 ， 该 表达 式 可 以 直接 是 一 个 具体 的 值 也 可 以 是 通 
过 表达 式 得 到 的 一 个 值 。 但 是 ， 这 个 值 必须 要 与 该 字段 的 数据 类 型 相 匹配 。 

上 面 的 语法 中 ， 只 是 给 表 中 的 一 个 字段 设置 了 默认 值 约 束 。 也 可 以 为 表 中 的 多 个 字段 

同时 设置 默认 值 约束 。 有 一 个 问题 请 读者 一 定 要 记 住 ， 那 就 是 每 一 个 字段 都 只 能 设置 一 个 

默认 值 约束 。 


全 说 明 : 有 两 个 类 型 的 列 是 不 能 够 为 其 创建 默认 值 约束 的 ， 一 个 是 timestamp 类 型 的 列 
一 个 是 具有 IDENTITY 属性 的 列 。 


有 了 设置 默认 值 的 语法 ， 就 让 我 们 演练 一 下 吧 。 

【示例 12】 在 创建 水 果 信 息 表 (fruitinfo〉 时 为 水 果 产 地 列 (origin) 添加 一 个 默认 值 
“海南 ”。 水 果 信 息 表 (fruitinfo〉 的 表 结 构 如 表 4-2 所 示 。 

根据 在 创建 表 时 添加 默认 值 的 语法 , 创建 水 果 产 地 列 的 默认 值 为 “海南 ”的 语句 如 下 : 


CREATE TABLE fruitinfo 
( 
id INT PRIMARY KEY, 
name VARCHAR(20), 
price DECIMAL(6,2), 
origin VARCHAR(20) DEFAULT ' 海 南 '， 
tel VARCHAR(15), 
remark VARCHAR (200) 
4 


由 于 商品 产地 列 的 数据 类 型 是 VARCHAR(20)， 默 认 值 “海南 ”满足 该 数据 类 型 要 求 。 
通过 上 面 的 语句 就 可 以 为 水 果 信 息 表 中 水 果 产 地 列 


aMicrosoft SQL Se .acenel ‘=olxl 

设置 默认 值 约束 。 执 行 效果 如 图 4.19 所 示 。 三 

如 果 读 者 在 运行 上 面 的 语句 时 也 出 现 了 图 4.19 es 忆 |D| 避 加 

2 | cter4 "A 行 WD) 》 曲名 

所 示 的 效果 , 那么 就 可 以 通过 sys.objects 来 查看 到 为 i Ss 
水 果 信 息 表 创建 的 默认 值 约束 了 。 Ss 


4.4.2 ”在 修改 表 时 添加 默认 值 约束 canl29) ppmmy 济南 


与 前 面 讲 过 的 主键 约束 和 外 键 约束 一 样 ， 默 认 | 
值 约束 也 可 以 在 修改 表 时 再 添加 。 但 是 ， 不 能 给 同 。 | 瑟 


一 个 列 再 添加 默认 值 约束 了 。 在 修改 表 时 添加 默认 上 
值 约束 ， 通 过 ALTER TABLE 语句 就 可 以 完成 。 具 二 多 
体 的 语句 如 下 : 图 4.19 在 建 表 时 给 水 果 产 地 添加 默认 值 


。86 。 


第 4 章 约束 表 中 的 数据 


ALTER TABLE table _ name 

ADD CONSTRAINT default name DEFAULT constant expression FOR col name; 

其 中 : 

口 table name: 表 名 。 它 是 要 创建 默认 值 约束 列 所 在 的 表 名 。 

口 default_ name: 默认 值 约束 的 名 字 。 该 名 字 可 以 省 略 ， 省 略 后 系统 将 会 为 该 默认 值 
约束 自动 生成 一 个 名 字 。 系统 自动 生成 的 默认 值 约束 名 字 通 常 是 df 表 名 _ 列 名 _ 随 
机 数 这 种 格式 的 。 

口 DEFAULT: 默认 值 约束 的 关键 字 。 如 果 省 略 默认 值 约束 的 名 字 ， 那 么 DEFAULT 
关键 字 直 接 放 到 ADD 后 面 ， 同 时 去 掉 CONSTRAINT。 

口 constant_expression: 常量 表达 式 ， 该 表达 式 可 以 直接 是 一 个 具体 的 值 也 可 以 是 通 
过 表达 式 得 到 的 一 个 值 。 但 是 ， 这 个 值 必 须要 与 该 字段 的 数据 类 型 相 匹 配 。 

口 col_name: 设置 默认 值 约束 的 列 名 。 


全 注意 ;' 当 不 设置 默认 值 约束 名 称 时 ， 在 修改 表 时 添加 默认 值 约束 的 语法 可 修改 为 如 下 
形式 。 


ALTER TABLE table name 
ADD CONSTRAINT default name DEFAULT constant expression FOR col name; 


-设置 默认 值 约束 

看 到 了 上 面 的 语法 ， 相 信 读 者 已 经 迫不及待 地 想 试 一 下 如 何在 修改 时 创建 默认 值 约束 
了 。 还 有 ， 如 果 在 已 经 创建 默认 值 约束 的 列 再 添加 默认 值 约束 会 发 生 什么 事情 呢 ?” 下 面 就 
分 别 来 演练 吧 。 

【示例 13】 给 水 果 信息 表 (fruitinfo) 中 的 备注 列 (remark) 添加 默认 值 约束 ， 将 其 默 
认 值 设置 成 “保质 期 为 1 天 ”。 

由 于 水 果 信 息 表 中 的 备注 列 没有 添加 过 默认 值 约束 ， 并 且 该 列 的 数据 类 型 是 
varchar(200)， 因 此 ， 设 置 默认 值 “ 保 质 期 为 1 天 ”是 满足 要 求 的 。 具 体 的 设置 语句 如 下 : 

ALTER TABLE fruitinfo 


ADD CONSTRAINT df fruitinfo remark DEFAULT ' 保 质 期 为 1 天 ' FOR remark; 
-- 为 remark 列 设置 默认 值 约束 


执行 上 面 的 语句 后 ， 就 可 以 为 水 果 信息 表 (fruitinfo) 的 备注 列 (remark) 添加 一 个 名 
为 df_fruitinfo_remark 的 默认 值 约束 了 。 效 果 如 图 4.20 所 示 。 


=|9jxl 


EE 
Wy Wa | chapter -| ?执行 DO 》 sv 吕 导 图 3 赵 | 难 圈 把 | 三 全 


SQLQuery2. s... ator 
ALTER TABLE Er: 
LADD CONSTRAINT 


| 
oe 


加 查询 已 成 功 执 … | WIDW-9527\MISSQLSERYER2008 (| WTDW-9527\Adninistrato..，| chspter4 |00:00:00 |0 行 
就 绪 行 2 列 70 Ch 65 3 


图 4.20 为 水 果 信息 表 中 的 备注 列 添加 默认 值 
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【示例 14】 给 水 果 信 息 表 (fruitinfo〉 中 的 备注 列 (remark) 再 添加 一 次 默认 值 约束 ， 
将 其 默认 值 设置 成 “保质 期 为 2 天 ”。 

由 于 水 果 信 息 表 (fruitinfo〉 中 的 备注 列 (remark) 已 经 被 示例 13 添加 了 一 个 默认 值 
约束 了 ， 那 么 再 添加 一 个 默认 值 约束 会 发 生 什么 呢 ? 添 加 默认 值 约束 的 语句 如 下 : 


ALTER TABLE fruitinfo 
RDD CONSTRAINT df fruitinfo remarkl DEFAULT ' 保 质 期 为 2 天 ' FOR remark; 


执行 上 面 的 语句 后 ， 会 发 生 如 下 问题 ， 如 图 4.21 所 示 。 


Ricrosoft SQL Server Nanagenent Studio 
文件 四 ”编辑 台 )， 视图 W)， 查询 @) 项 目 @) 调 趟 W) 工具 G) 窗口 如 ”社区 帮助 吕 
辽 新 建 查询 名 | 出 | 也 也 吕 | 马 | 区 回忆 | 过 

9 HR | chepter4 -| 了 扫 行 Wb》 v 品 晤 国 | 玉 绚 | 镁 圈 扩 | 三 全 名 
“SQLQuery2. s... ator (51))*| -x 


日 ALTER TABLE fruitinfo 
上 apD CONSTRAINT af_fruicinfo_remarkl DEFAULT ' 保 质 期 为 2 天 '| FOR remark; 


已 消息 


消息 | 级 别 16， 状 态 5 第 1 行 下 

已 在 列 上 乡 定 了 pgpAUL 中 

消息 1750， 级 别 16， 状 态 o， 第 1 行 

无 法 创建 约束 。 请 参 赔 前 面 的 错误 消息 。 vi 
J » 


心 查 询 已 完成 ，… | WIDW-9527 WSSQLSERVER2008 〔 . ，| WIDN-9527\Adninistrato. .，| chspter4 |00:00:00 |0 行 
职 绪 行 2 列 59 Ch 54 


图 4.21 为 一 个 列 添加 多 个 默认 值 约束 的 错误 提示 
从 图 4.21 中 就 可 以 看 出 ， 给 表 中 的 每 一 个 列 只 能 添加 一 个 默认 值 约束 。 
4.4.3 ”去 除 默 认 值 约束 


当 表 中 的 某 个 字段 不 再 需要 默认 值 时 ， 去 除 默 认 值 约束 是 非常 容易 的 。 有 的 读者 会 想 
到 ， 直 接 将 默认 值 变 成 是 NULL 不 就 可 以 了 吗 ? 其 实 ， 这 个 想法 是 很 好 的 ， 但 是 不 可 能 成 
功 。 原 因 就 是 前 面 在 添加 默认 值 时 说 过 ， 一 个 列 的 只 能 有 一 个 默认 值 ， 已 经 设置 了 默认 值 
的 列 就 不 能 够 再 重新 设置 了 。 如 果 想 重新 设置 也 只 能 先 将 其 默认 值 删除 ， 然 后 再 添加 。 因 
此 ， 当 默认 值 不 再 需要 时 ， 只 能 将 其 删除 掉 。 删 除 默认 值 约束 的 语法 如 下 : 

ALTER TABLE table name 

DROP CONSTRAINT df name; 

这 里 ，df name 就 是 默认 值 约束 的 名 字 。 

读者 从 上 面 的 语法 中 可 以 看 出 , 删除 约束 的 方法 都 很 相似 , 只 是 约束 的 名 字 不 同 而 已 。 
下 面 就 请 读者 运用 上 面 的 语法 完成 示例 15 的 练习 。 

【示例 15】 将 水 果 信 息 表 〈fruitinfo〉 中 添加 的 名 为 df_fruitinfo_remark 的 默认 值 约束 
删除 。 

在 本 示例 中 已 经 清楚 地 知道 了 要 删除 默认 值 约束 的 名 字 ， 如 果 不 清楚 要 删除 的 默认 值 
约束 名 字 就 需要 使 用 SYS.OBJECTS 先 查 询 一 下 了 。 删除 df fruitinfo remark 默认 值 约束 的 
语句 如 下 : 

ALTER TABLE fruitinfo 

DROP CONSTRAINT df fruitinfo remark; =-- 删 除 默 认 值 约束 
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执行 上 面 的 语句 后 ， 就 可 以 将 默认 值 约束 df_fruitinfo_remark 从 fruitinfo 表 中 移 除 了 。 
效果 如 图 4.22 所 示 。 


zosoft SQL Server anecenendl i -TE 
文件 中 编辑 下 视图 VW 查 淘 @) 项 目 E) 调式 W) 工具 中 
窗口 ) 社区 (下 助 如 
卫 亲 建 查询 外 | 让 | 也 也 可 | 记 | 蕊 台 本 | 双 忆 
CT | hapters -| 执行) 》 和 v 中 旺 国 | 时 
‘SQLQuery2. s- - - ater (51))#* 
日 ALTER TABLE fruitinfo 
上 -DROP CONSTRAINT df fruicinfo_rermarK: 


已 消息 | 
命令 已 成 功 完成 。 


hssqrsERvER2009 ( | TDw-9s27\WAdninistrate .. | chapter4 |00:00:00 |0 行 
就 绪 行 3 列 ! 


图 4.22 ”删除 默认 值 约束 df_fruitinfo remark 的 效果 


Chl 


4.4.4 ”使 用 企业 管理 器 轻松 使 用 默认 值 约束 


在 企业 管理 器 中 添加 和 删除 默认 值 约束 是 很 简单 的 ， 只 要 记 住 给 列 添加 默认 值 约束 时 
要 使 默认 值 与 列 的 数据 类 型 匹配 ， 如 果 是 字符 类 型 的 还 要 加 上 单 引号 。 下 面 就 将 示例 12、 
示例 13 以 及 示例 15 在 企业 管理 器 中 演练 一 下 。 

【示例 16】 在 企业 管理 器 中 ， 创 建 水 果 信息 表 (fruitinfo〉 并 在 水 果 产 地 列 (origin) 
添加 默认 值 “ 海 南 ”。 

在 企业 管理 器 中 ， 创 建 表 的 同时 添加 默认 值 约束 的 步骤 分 为 如 下 3 步 。 

(1) 打开 创建 表 界 面 

在 对 象 资源 管理 器 中 , 展开 要 创建 数据 表 的 数据 库 节 点 , 并 右 击 该 数据 库 下 的 表 节 点 ， 
在 弹出 的 右键 菜单 中 选择 “新 建 表 ”选项 ， 界 面 如 图 4.23 所 示 。 

(2) 录入 水 果 信 息 表 的 列 信息 

在 图 4.23 所 示 的 界面 中 ， 录 入 表 4-2 所 示 的 水 果 信 息 表 的 列 信息 并 将 表 保 存 成 
fruitinfo。 录 入 后 的 效果 如 图 4.24 所 示 。 


Ricrosoft SQL Server Wanaceseatl Se I Ed| 
文件 四 ”编辑 有 视图 WD 项目) 调式 0) 


忽 辑 EF) 视图 顶 
表 设 计 器 LL) 工具 CD) 窗口 WD 社区 C) 帮助 0 素 设 计 器 (工具 人 窗口) 社区 CC) 帮助 和 0 
也 新建 查询 中 | 记 | 亿 也 配 | 记 | 蕊 轩 局 | 运 后 也 新建 查询 岂 | 让 | 饭 也 马上 记 | 启 央 与 
霹 | 7 123 音 讽 遍 回 园 司 OE 


| YIDY-9527\0. .. dbo. Table_2 


| YIDY-9527W. .0. fruitinfo 
数据 类 型 


图 4.23 表 的 设计 界面 图 4.24 水 果 信息 表 (fruitinfo) 的 列 信息 
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(3) 设置 默认 值 

根据 题目 要 求 ， 要 对 水 果 产 地 列 设置 默认 值 “ 海 南 ”。 在 图 4.24 所 示 的 界面 中 选择 水 
果 产 地 (origin〉 列 ， 并 展开 列 属性 界面 ， 如 图 4.25 所 示 。 

在 图 4.25 所 示 的 界面 中 ， 在 “默认 值 或 绑 定 ”选项 后 面 添加 上 “海南 ”作为 默认 值 。 
效果 如 图 4.26 所 示 。 


ie =I9|x| . 
i pe ET 和 调试 四 ) 表 设 计 器 册 ) Microsoft erver Banagenent Studio =I9|x| 
工具 外 窗口 如 社区 C) 帮助 0 文件 中 ”编辑 EE) 视图 YY) 项目) 调式 表 设 计 器 已) 


工具 中 窗口 WD 社区 GC 帮助 中 
| 新建 查 询 虽 | 站 信访 本 ~ 
A 2Wessw | 入 | 三 十 久 D|S 晶 | 加 
- a 电 |? | 吗 广 册 册 回 罚 名 


| YIDY-9527\0. .. 0. fruitinfo# vx 二 [TEFS x 
一 区 2 info Ey 
列 各 数据 类 型 。 | 多 许 Nu 人 本 光 洗 Nl 从 


日 加 加 
varchar(20) 区 
(20) 
decmal(s, 2) 反 Te 吕 
| 了 | 7 

varchar(15) a varchar(15) 区 
varchar(200) 区 varchar(200) Ld 

[el 品 


图 4.25 水 果 产 地 列 〈origin) 的 列 属性 界面 图 4.26 设置 默认 值 界面 
在 图 4.26 所 示 的 界面 中 ， 单 击 “ 保 存 ” 的 图 标 回 即 可 完成 默认 值 的 添加 操作 。 


外 说 明 ， 在 企业 管理 器 中 给 表 中 列 设置 默认 值 时 ， 可 以 对 字符 串 类 型 的 数据 省 略 单 引号 ， 
如 果 省 略 了 单 引号 ， 系 统 会 在 保存 表 信息 时 自动 为 其 加 上 单 引号 的 。 


【示例 17】 在 企业 管理 器 中 ， 为 水 果 信 息 表 (fruitinfo) 中 的 备注 列 (remark) 添加 默 
认 值 约束 ， 将 其 默认 值 设置 成 “保质 期 为 1 天 ”。 

有 了 示例 16 的 基础 ， 修 改 表 的 时 候 再 添加 默认 值 就 可 以 省 去 创建 表 的 步骤 ， 直 接 进 
入 到 表 的 设计 页 面 添 加 默认 值 就 可 以 了 。 在 水 果 信息 表 (fruitinfo ) 中 为 备注 列 (remark ) 
添加 默认 值 约束 ， 可 以 分 为 如 下 两 个 步骤 。 

(1) 打开 水 果 信 息 表 的 表 设 计 界 面 

在 企业 管理 器 中 的 对 象 资源 管理 器 里 ， 右 击 水 果 信 息 表 (fruitinfo)， 在 弹出 的 右键 菜 
单 中 选择 “设计 ”选项 ， 即 可 进入 水 果 信 息 表 的 设计 页 面 ， 与 图 4.24 一 致 。 

(2) 添加 默认 值 约束 

在 水 果 信 息 表 (friutinfo) 的 设计 界面 里 , 选择 备注 列 (remark) 并 在 其 列 属 性 中 的 “ 默 
认 值 或 绑 定 ” 选 项 后 面 ， 加 上 “保质 期 为 1 天 ”的 默认 值 。 效 果 如 图 4.27 所 示 。 

现在 已 经 大 功 告 成 了 ， 就 差 保 存 了 。 在 图 4.27 所 示 的 界面 中 , 单 击 “ 保 存 ” 的 图 标 加 
即 可 完成 默认 值 的 添加 操作 。 
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【示例 18】 在 企业 管理 器 中 ， 删 除 水 果 信息 表 (fruitinfo〉 中 备注 列 的 默认 值 约束 。 

在 企业 管理 器 中 ， 删 除 默认 值 与 添加 默认 值 很 相像 ， 只 是 在 删除 的 时 候 将 默认 值 清 空 
就 可 以 了 。 当 然 ， 删 除 默认 值 的 操作 也 要 在 水 果 信 息 表 的 设计 界面 来 完成 。 同 样 ， 也 分 为 
两 个 步骤 。 

(1) 打开 水 果 信 息 表 的 设计 页 面 

在 企业 管理 器 中 的 对 象 资源 管理 器 里 ， 右 击 水 果 信 息 表 (fruitinfo)， 在 弹出 的 右键 菜 
单 中 选择 “设计 ”选项 ， 即 可 进入 水 果 信 息 表 的 设计 页 面 ， 与 图 4.24 一 致 。 

(2) 去 除 默认 值 

在 企业 管理 器 中 去 除 默认 值 时 ， 只 需要 将 要 去 除 默认 值 的 列 的 默认 值 清空 即 可 。 将 水 
果 信息 表 (fruitinfo) 中 备注 列 (remark) 中 默认 值 清空 的 效果 如 图 4.28 所 示 。 


Ricrosoft SQL Server Wanagenent S130 | Ricrosoft SQL Server Nanseeses@ St 
文件 到 ) 编辑 EE) 视图 YW) 项 目下 ) 调式 ) 甫 设计 器 册 ) 文件 他) 编辑 E) 视图 WD 项 目 f) 调试 ) 表 设计 器 几 ) 
工具 窗口 轨 社区 C) 帮助 0 


工具 人 窗口 如 社区 C) 帮助 o 


卫 犁 建 查询 思 | 岂 | 也 也 加 | 记 | 区 呈 忆 | 运 下 卫 章 建 查询 中 | 记 | 轧 也 下 外 | 蕊 回 己 | 又 瑟 


二 |? | 双亲 见 出 四 加 轧 |? | 蕊 订 册 吗 回 时 
TIDY-9527\0. .. 0. fraitinfo* vx SQLQuery3. s... ator (55))# | zx 
列 名 数据 类 型 允许 Nul 值 列 名 数据 类 型 次 许 Nl 值 
yd int [a yd int rc 
name varchar(20) 区 可 name varchar(20) 区 
price decimal(6, 2) 区 Price decimak6, 2) 5 
origin varchar(20) La orign varchar(20) a 
tel varchar(15) 区 ta varchar(15) 区 
UL varchar(200) a varchar(200) a 
口 [本 


图 4.27 给 备注 列 〈remark) 添加 默认 值 图 4.28 ”去除 水 果 信 息 表 中 备注 列 的 默认 值 


一 定 还 要 记 住 ， 清 空 默认 值 后 ， 还 要 保存 ! 
通过 上 面 的 3 个 示例 ,相信 读者 就 可 以 完全 掌握 在 企业 管理 器 中 使 用 默认 值 的 方法 了 。 
任何 一 种 方法 只 要 多 加 练习 ， 一 定 会 牢 牢 掌握 的 。 


4.5 检查 约束 一 一 CHECK 
所 谓 检 查 约 束 ， 从 字面 上 的 意思 理解 就 是 用 来 对 数据 进行 检查 的 。 质 量 检查 员 都 听 说 
过 吧 ， 在 每 件 商品 出 三 前 都 会 对 商品 的 各 种 标准 进行 核对 ， 核 对 正确 后 才能 够 将 检验 合格 


的 标签 贴 到 商品 上 。 贴 了 检验 合格 的 标签 后 才能 够 正常 地 流通 到 市 场 上 。 检 查 约束 的 作用 
就 是 为 了 确保 数据 表 添 加 的 数据 是 有 效 的 ， 在 添加 之 前 对 数据 的 一 种 检查 。 
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4.5.1 在 建 表 时 添加 检查 约束 


检查 约束 在 一 张 数据 表 中 可 以 有 多 个 ， 但 是 每 一 列 只 能 设置 一 个 检查 约束 。 虽 然 检查 


约束 可 以 帮助 数据 表 检 查 数据 确保 数据 的 正确 性 ,但 是 也 不 能 给 每 一 个 列 都 设置 检查 约束 ， 
否则 ， 就 会 影响 数据 表 中 数据 操作 的 效率 。 因 此 ， 在 给 表 设 置 检查 约束 前 ， 也 要 尽 可 能 
确保 设置 检查 约束 的 列 是 否 真 的 有 用 。 


说 了 这 么 多 ， 相 信 读 者 已 经 迫不及待 地 想 知道 如 何 设置 检查 约束 了 。 在 建 表 时 就 可 以 


同时 将 检查 约束 设置 好 ， 这 样 也 省 去 了 以 后 设置 的 麻烦 。 建 表 时 添加 检查 约束 的 语法 有 两 
种 形式 。 不 管 使 用 哪 种 形式 ， 请 读者 记 住 了 检查 约束 的 关键 字 是 CHECK。 


1. 使 用 SQL 语句 设置 列 级 检查 约束 


CREATE TABLE NAME table name 

( 

COLUMN NAME1 DATATYPE CHECK (expression), 
COLUMN NAME2 DATATYPE, 

COLUMN NAME3 DATATYPE, 


其 中 : 

口 CHECK: 检查 约束 的 关键 字 。 

口 expression: 约束 的 表达 式 ， 可 以 是 1 个 条 件 也 可 以 同时 有 多 个 条 件 。 比 如 : 设置 
该 列 的 值 大 于 10， 那 么 ， 表 达 式 就 可 以 写成 COLUMN_NAME1>10;， 设置 该 列 的 
值 在 10 一 20 之 间 ， 那 么 ， 表 达 式 就 可 以 写成 COLUMN NAME1>10 and 
COLUMN_NAME1<20。 检查 约束 全 是 靠 表 达 式 来 进行 数据 检查 的 。 因 此 ,表达 式 
的 编写 是 最 重要 的 ， 请 读者 多 加 练习 。 


2. 使 用 SQL 语句 设置 表 级 检查 约束 


CREATE TABLE NAME table name 
( 

COLUMN NAME] DATATYPE 

COLUMN NAME2 DATATYPE, 
COLUMN NAME3 DATATYPE, 


CONSTRAINT ck name CHECK (expression), 

CONSTRAINT ck name CHECK (expression), 

其 中 : 

口 ck_name: 检查 约束 的 名 字 。 它 要 写 在 CONSTRAINT 关键 字 的 后 面 ， 并 且 检 查 约 
束 的 名 字 不 能 重 名 。 检查 约束 的 名 字 通 常 是 以 ck 开头 的 。CONSTRAINT ck_name 
部 分 省 略 后 ， 系 统 会 自动 为 检查 约束 设置 一 个 名 字 ， 系 统 设置 的 名 字 通 常 是 “ck_ 
表 名 _ 列 名 _ 随 机 数 ”的 形式 。 

口 CHECK(expression): 检查 约束 的 定义 。 

了 解 了 上 面 的 两 种 语法 形式 ， 就 可 以 很 容易 地 为 表 创 建 检查 约束 了 。 下 面 就 来 演练 这 
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两 种 添加 检查 约束 的 语法 吧 。 
【示例 19】 在 创建 水 果 信 息 表 (fruitinfo) 时 ， 给 水 果 价 格 列 〈price) 添加 检查 约束 。 
要 求 水 果 的 价格 都 要 大 于 0 元 。 
下 面 使 用 添加 检查 约束 的 两 种 方法 ， 分 别 在 创建 水 果 信 息 表 时 给 水 果 价 格 列 添加 检查 
约束 。 
(1) either nie 
创建 表 并 添加 检查 约束 ， 语 句 如 下 : 


CREATE TABLE fruitinfo 
( 
id INT PRIMARY KEY, 
name VARCHAR(20), 
price DECIMAL(6,2) CHECK (price>0), 
origin VARCHAR(20), 
tel VARCHAR (15), 
remark VARCHAR(200) 
) 


执行 上 面 的 语法 ， 就 可 以 为 水 果 信 息 表 (fruitinfo) 中 的 水 果 价格 列 (price) 添加 检查 
约束 。 以 后 再 向 该 列 输入 值 时 ， 都 必须 要 大 于 0。 

(2) 使 用 在 表 级 添加 检查 约束 的 语法 

创建 表 并 添加 检查 约束 ， 语 句 如 下 : 


CREATE TABLE fruitinfo 

( 
id INT PRIMARY KEY, 
name VARCHAR(20), 
Price DECIMAL(6,2), 
origin VARCHAR(20), 
tel VARCHAR (15), 
remark VARCHAR(200), 
CHECK (price>0) 

) 


执行 上 面 的 语句 ， 同 样 可 以 为 水 果 信 息 表 (fruitinfo〉 中 的 水 果 价 格 列 (price) 添加 检 
查 约束 。 但 是 ， “i | 个 问题 读者 要 注意 ， 在 micreseft SQL Server Nanseeneatl Se 
用 第 1 种 方法 创建 表 并 添加 检查 约束 后 ， 不 A WA TAO 
能 直接 再 使 用 第 2 种 方法 创建 表 了 ， 否 则 ， 了 2 加 | 记 上马 甩 及 | 应 | 不 同 所 | 承 司 


EE- | 执行 &) 》 w ?3 


就 会 出 现 该 表 已 经 存在 的 错误 提示 。 因此， ez 3 

人 成 了 入 | 入 法 后 ， 读 要 和 水。 | 

果 信 息 表 删除 ， 然 后 再 用 第 2 种 方法 创建 表 manceo), 

添加 检查 约束 。 效 果 如 图 4.29 所 示 。 ee es ol 

4.5.2 在 修改 表 时 添加 检查 约束 
如 果 在 创建 表 时 没有 直接 添加 检查 约 | 并 ns. 司 

束 ， 也 可 以 在 修改 表 的 时 候 来 添加 检查 约束 。 Hs [5 -|YTDY-9s2TWhdaninistrate .. | chapter4 |00:00:00 大 

在 修改 表 时 只 能 给 没有 添加 检查 约束 的 列 添 Wm 7 NH mH 


加 检查 约束 。 修 改 表 时 添加 检查 约束 也 是 通 。 图 4.29 创建 水 果 信 息 表 并 为 价格 添加 检查 约束 


.93 。 
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过 使 用 ALTER TABLR 语句 来 完成 的 ， 记 住 下 面 的 语法 形式 ， 你 就 可 以 完成 在 修改 表 时 添 
加 检查 约束 的 操作 了 。 
ALTER TABLE table name 
ADD CONSTRAINT ck name CHECK (expression); 
只 中 : 
口 table name: 表 名 。 
口 CONSTRAINT ck_name: 添加 名 为 ck_name 的 约束 。 该 语句 可 以 省 略 ， 省 略 后 系 
统 会 为 添加 的 约束 自动 生成 一 个 名 字 。 
口 CHECK(expression): 检查 约束 的 定义 。CHECK 是 检查 约束 的 关键 字 ，expression 
是 检查 约束 的 表达 式 。 

下 面 就 将 示例 19 中 添加 的 检查 约束 在 创建 水 果 信 息 表 后 添加 , 请 看 示例 20 的 操作 吧 ! 

【示例 20】 先 按 照 表 4-2 的 要 求 创建 水 果 信息 表 〈fruitinfo)， 然 后 再 给 水 果 价格 列 
(price) 添加 检查 约束 。 要 求 水 果 价 格 大 于 0。 

创建 水 果 信 息 表 的 语句 在 示例 1 中 就 已 经 提 及 过 了 ， 这 里 就 不 再 演示 了 。 下 面 就 直接 
为 水 果 信 息 表 (fruitinfo〉 中 的 水 果 价 格 列 (price》 添 加 检查 约束 ， 语 句 如 下 : 

ALTER TABLE fruitinfo 

ADD CONSTRAINT ck fruitinfo price CHECK (price>0); 

这 里 ， 在 添加 检查 约束 时 ， 给 其 检查 约束 命名 为 ck_fruitinfo_price 。 也 可 以 省 略 
CONSTRAINT ck_fruitinfo_price 部 分 不 直接 给 检查 约束 命名 ， 而 是 由 系统 自动 为 其 命名 。 
执行 上 面 的 语句 就 可 以 为 水 果 信 息 表 (fruitinfo) 中 的 水 果 价 格 列 (price) 添加 检查 约束 了 。 
效果 如 图 4.30 所 示 。 

RMicrosoft SQL Server Nanagenent Studio =I9|xl 


文件 中 编辑 下 ) 视图 WW) 查询) 项 目 @) 调式 W) 工具 YI) 窗口 如 
社区 CC) 帮助 00 

辽 新 建 查 询 W | 六 | 六 迎 串 | 书 | 蕊 回 马 加: 

Wy Wo | cheter4 “| ?执行 0 有 v 中 地 加 | 守 艳 号 
”SQLQuery4. s. . - ater (56))*| WIIDY-S52TWL 0 fruitinfo# | 


章 ALTER TABLE fruitinfo 
LADD CONSTRAINT ck fruitinfo price CHECK(price>0); 


耳 消 息 | 
命令 已 成 功 完成。 


il 
-9527\NSSQLSERVER2008 〔 .| YIDN-9527\Adninistrato. . .| chapter4 |00:00:00 |0 行 
就 绪 行 2 列 50 Ch 50 


图 4.30 修改 水 果 信息 表 时 给 价格 列 添加 检查 约束 


4.5.3 ”去除 检查 约束 


检查 约束 同 前 面 讲解 过 的 其 他 约束 一 样 ， 都 是 不 能 够 直接 修改 的 。 读 者 要 想 更 改 某 一 
列 的 检查 约束 ， 也 是 要 先 删除 该 检查 约束 ， 然 后 再 为 其 重新 创建 检查 约束 。 因 此 ， 删 除 检 
查 约束 的 语法 是 至 关 重要 的 ， 但 是 ， 也 与 其 他 的 约束 删除 类 似 。 具 体 的 语法 形式 如 下 : 


ALTER TABLE table name 
DROP CONSTRAINT ck name; 
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其 中 : 

口 table name: 表 名 。 

口 ck_name: 检查 约束 的 名 字 。 

相信 上 面 的 语法 形式 ， 读 者 已 经 不 陌生 了 ， 已 经 在 删除 约束 部 分 出 现 过 多 次 了 。 下 面 
就 利用 这 个 语法 来 完成 示例 21 。 

【示例 21】 删除 在 示例 20 中 为 水 果 信 息 表 (fruitinfo) 中 水 果 价格 列 〈price) 添加 的 
检查 约束 ck_fruitinfo price。 

知道 了 约束 的 名 字 还 有 其 所 在 的 表 ， 删 除 约束 就 是 轻而易举 的 事 了 。 删 除 检查 约束 的 
语句 如 下 : 

ALTER TABLE fruitinfo 

DROP CONSTRAINT ck fruitinfo price; -删除 检查 约束 


执行 上 面 的 语句 ， 就 可 以 将 水 果 信 息 表 中 的 水 果 价 格 列 的 检查 约束 删除 了 。 效 果 如 
图 4.31 所 示 。 Microsoft SQL Server Wanagenentl Se =Io|x| 
当然 ， 如 果 你 还 不 知道 要 删除 的 检查 约束 交加 视图 WW 查询 @) 项 目 @) 调 坛 0) 工具 中 


Se a i 有 窗口 社区 帮助 
叫 什么 名 ,那么 ， 还 是 按 老 规矩 通过 sys.objects 。 amaaam | 尸 杞 马 世 | 加 日 台 | 加 s 


来 查看 一 下 吧 ! 时 | seeer4 -26 Pv 刘 
ER TaLE Tract 
4.5.4 使 用 企业 管理 器 轻松 使 用 检查 上 pRop CONSTRAINT ck_truicinfo_price; 
约束 Sal 
DE 二 
通过 企业 管理 器 来 操作 检查 约束 也 是 很 简 。 onem e nme [Aw [mm 
2 1 chl > 


单 的 一 件 事 ， 可 以 说 有 了 前 面 的 基础 ， 读 者 面 
对 这 个 问题 是 小 菜 一 碟 的 事 。 怎 么 使 用 企业 管 
理 器 来 操作 检查 约束 呢 ? 仍然 是 按 老 办 法 , 重 写 示例 19 一 示例 21。 如 果 读 者 有 足够 的 信心 ， 
可 以 自己 先 来 尝试 一 下 ， 如 果 遇 到 问题 了 再 来 查阅 下 面 的 答案 。 

【示例 22】 在 企业 管理 器 中 ， 创 建 水 果 信息 表 时 为 水 果 价 格 列 添加 检查 约束 ， 要 求 水 
果 价 格 要 大 于 0 元 。 

如 果 要 在 企业 管理 器 中 在 创建 表 的 同时 为 表 中 的 列 添加 检查 约束 ， 可 以 通过 下 面 的 3 
个 步骤 完成 。 实 际 上 ， 读 者 也 可 以 根据 前 面 在 企业 管理 器 中 操作 约束 的 经 验 ， 自 己 摸索 一 
下 添加 检查 约束 的 方法 。 

(1) 打开 创建 表 的 界面 并 录入 表 的 信息 

在 企业 管理 器 的 对 象 资源 管理 器 中 ， 展 开 要 创建 数据 表 的 数据 库 节点 ， 并 右 击 该 数据 
库 下 的 表 节 点 ， 在 弹出 的 右键 菜单 中 选择 “新 建 表 ” 选 项 ， 即 可 打开 创建 表 的 界面 。 与 
图 4.23 一 致 。 根 据 表 4-2 所 示 的 水 果 信 息 表 的 结果 录入 表 的 信息 。 

(2) 打开 CHECK 约束 的 界面 

右 击 水 果 信息 表 的 设计 界面 ， 在 弹出 的 右键 菜单 中 选择 “CHECK 约束 ”， 出 现 图 4.32 
所 示 的 界面 。 

从 图 4.32 所 示 的 界面 中 ， 可 以 看 出 水 果 信息 表 (fruitinfo) 中 所 有 的 存在 的 检查 约束 。 
目前 该 图 显示 的 结果 是 当前 没有 检查 约束 。 


图 4.31 删除 价格 列 的 检查 约束 
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选 定 的 CHECK 约束 @) 


区 we 


添加 心 ) sa 


图 4.32 ” CHECK 约束 界面 


(3) 添加 检查 约束 
在 图 4.32 所 示 的 界面 中 ， 单 击 “ 添 加 ”按钮 ， 出 现 图 4.33 所 示 界 面 。 
了 可 


选 定 的 CHECK 约束 8); 
ReitinEo 


(名 称 ) CRK_fruitinfo 
说 明 
日 表 设计 器 
强制 用 于 INSERT 和 UPDA 是 
制 用 于 复制 是 
在 凶 建 或 重新 启用 时 检查 下 是 
CE |]_ wmow | 关闭 器 


图 4.33 ”添加 检查 约束 界面 


在 图 4.33 所 示 的 界面 中 ， 在 “表达 式 ” 选 项 后 面 填 入 商品 价格 大 于 0 的 检查 约束 。 效 
果 如 图 4.34 所 示 。 
可 可 


选 定 的 CHECK 约束 (8): 
CK_fruitinfo# 


强制 用 于 INSERT 和 UPDX 是 
强制 用 于 复制 是 
在 创建 或 重新 启用 时 检查 丈 是 


图 4.34 添加 检查 约束 后 的 效果 
在 图 4.34 所 示 的 界面 中 , 单 击 “关闭 ”按钮 并 保存 表 信息 ， 即 可 完成 检查 约束 的 添加 。 
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【示例 23】 在 企业 管理 器 中 ， 修 改 水 果 信 息 表 时 为 水 果 价 格 列 添加 检查 约束 ， 要 求 水 
果 价 格 要 大 于 0 元 。 

在 修改 水 果 信息 表 时 为 水 果 价 格 列 添加 检查 约束 ， 就 相当 于 是 从 示例 22 中 的 第 2 个 
步骤 开始 做 起 ， 分 为 两 个 步骤 就 可 以 完成 。 那 么 ， 读 者 不 能 怕 麻 烦 一 定 要 重新 练习 一 下 ， 
这 样 才 能 更 好 地 掌握 对 检查 约束 的 使 用 。 

(1) 打开 添加 检查 约束 的 界面 
展开 水 果 信息 表 的 设计 页 面 ， 并 右 击 该 界面 ， 在 弹出 的 右键 菜单 中 选择 “CHECEK 的 
束 ” 选 项 。 出 现 图 4.32 所 示 的 界面 。 

(2) 添加 检查 约束 

在 图 4.32 所 示 的 界面 中 ， 单 击 “ 添 加 ”选项 ， 在 该 界面 中 的 “表达 式 ” 选 项 后 面 填 入 
检查 约束 的 表达 式 。 效 果 与 图 4.34 一 致 。 添 加 检查 约束 后 ， 要 关闭 该 界面 。 

通过 上 面 的 两 个 步骤 就 可 以 完成 对 检查 约束 的 添加 ， 但 是 ， 也 不 忘记 保存 表 信 息 。 

【示例 24】 在 企业 管理 器 中 ， 删 除 水 果 信息 表 中 水 果 价 格 列 的 检查 约束 。 

删除 水 果 信息 表 中 水 果 价 格 列 的 检查 约束 很 简单 ， 与 前 面 讲 过 的 删除 外 键 约束 非常 相 
像 。 读 者 可 以 先 不 看 下 面 的 操作 ， 自 己 动手 操作 一 下 。 删 除 检查 约束 通常 可 以 分 为 两 个 步 
又 操作 。 

(1) 打开 表 的 检查 约束 界面 

表 的 检查 约束 界面 与 图 4.32 类 似 ， 目 前 水 果 信息 表 经 过 前 面 的 两 个 示例 的 操作 ， 只 有 
一 个 检查 约束 。 效 果 如 图 4.34 所 示 。 

(2) 删除 指定 的 检查 约束 

在 图 4.34 所 示 的 界面 中 ， 选 择 为 水 果 价 格 列 添加 的 检查 约束 CK_fruitinfo， 单 击 “ 删 
除 ”按钮 ， 即 可 删除 该 检查 约束 。 


4.6 唯一 约束 一 一 UNIQUE 


唯一 约束 的 名 字 看 起 来 就 很 霸道 吧 。 其 实 ， 它 也 是 名 副 其 实 的 霸道 约束 。 那 么 ， 它 是 
如 何 霸道 呢 ? 首先 ， 唯 一 约束 是 用 来 确保 列 中 值 的 唯一 性 ， 其 次 ， 它 还 能 够 同时 为 表 中 的 
多 个 列 设置 唯一 约束 。 在 什么 情况 下 ， 会 考虑 为 表 中 的 列 设置 唯一 约束 呢 ? 想 一 想 ， 如 果 
是 一 个 学 生 信 息 表 ， 需 要 确保 唯一 的 列 有 学 生 的 学 号 、 学 生 的 身份 证 号 以 及 学 生 的 图 书 卡 
号 等 信息 。 有 这 么 多 列 都 需要 确保 唯一 性 ， 可 见 唯一 约束 有 用 吧 。 


4.6.1 在 建 表 时 加 上 唯一 约束 


说 到 唯一 约束 ， 读 者 一 定 会 想起 来 主键 约束 也 可 以 确保 唯一 啊 ! 但 是 ， 主 键 约束 是 在 
一 个 表 中 只 能 有 一 个 的 ， 如 果 要 想 给 多 个 列 设置 唯一 性 ， 还 是 需要 使 用 唯一 约束 的 。 在 创 
建 表 时 就 可 以 直接 为 表 中 的 列 设置 唯一 约束 。 在 建 表 时 添加 唯一 约束 可 以 通过 下 面 两 种 语 
法 形式 来 完成 。 请 记 住 唯一 约束 的 关键 字 UNIQUE。 

1. 使 用 SQL 语句 设置 列 级 唯一 约束 


设置 列 级 的 唯一 约束 就 很 简单 了 , 只 要 在 列 的 数据 类 型 后 面 加 上 UNIQUE 关键 字 就 可 
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以 了 。 具 体 的 语法 如 下 : 


CREATE TABLE NAME table name 
( 

COLUMN NAME1 DATATYPE UNIQUE, 
COLUMN NAME2 DATATYPE, 
COLUMN NAME3 DATATYPE 


这 里 ， 就 是 给 COLUMN NAME1 列 设置 了 唯一 约束 ， 也 可 以 同时 给 多 个 列 设置 唯一 


约束 。 


2. 使 用 SQL 语句 设置 表 级 唯一 约束 
表 级 唯一 约束 的 创建 就 要 比 列 级 麻烦 一 些 , 但 是 也 不 是 很 难 的 。 表 级 唯一 约束 的 添加 ， 


还 是 在 所 有 列 定义 的 后 面 直接 添加 。 具 体 的 语法 如 下 : 


CREATE TABLE NAME table name 
( 

COLUMN NAME]1 DATATYPE, 
COLUMN NAME2 DATATYPE, 
COLUMN NAME3 DATATYPE 


CONSTRAINT uq name UNIQUE (col namel), 

CONSTRAINT uq name UNIQUE (col name2), 

其 中 : 

口 CONSTRAINT: 在 表 中 定义 约束 时 的 关键 字 。 

口 uq_name: 唯一 约束 的 名 字 。 唯 一 约束 的 名 字 可 以 省 略 ， 省 略 时 也 要 将 其 前 面 的 
CONSTRAINT 关键 字 一 并 省 略 。 如 果 省 略 了 唯一 约束 的 名 字 ， 系 统 会 为 其 自动 生 
成 一 个 “UQ _ 表 名 _ 随 机 数 ” 形 式 的 名 字 。 

口 UNIQUE(col name): UNIQUE 是 定义 唯一 约束 的 关键 字 ， 不 可 省 略 。col_name 是 
定义 唯一 约束 的 列 名 。 

有 了 上 面 的 两 种 语法 形式 ， 就 可 以 在 创建 表 时 添加 唯一 约束 了 。 至 于 用 哪 种 方法 全 和 赁 


读者 的 喜好 ， 但 是 在 学 习 的 时 候 还 是 两 种 方法 都 尝试 一 下 吧 。 


【示例 25】 分 别 使 用 上 面 的 两 种 语法 ， 在 创建 水 果 信息 表 (fruitinfo) 时 将 水 果 名 称 


列 (name) 设置 成 唯一 约束 。 


(1) 使 用 列 级 添加 唯一 约束 的 语法 
创建 表 并 添加 检查 约束 ， 语 句 如 下 : 


CREATE TABLE fruitinfo 

( 
id INT PRIMARY KEY, 
name VARCHAR(20) UNIQUE, 
price DECIMAL(6,2), 
origin VARCHAR(20), 
tel VARCHAR (15), 
remark VARCHAR(200) 
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执行 上 面 的 语法 ， 就 可 以 为 水 果 信 息 表 (fuitinfo) 中 的 水 果 名 称 列 name) 添加 唯 
一 约束 。 

(2) 使 用 表 级 添加 检查 约束 的 语法 
创建 表 并 添加 检查 约束 ， 语 句 如 下 : 
CREATE TABLE fruitinfo 
( 

id INT PRIMARY KEY, 

name VARCHAR(20), 

price DECIMAL(6,2), 

origin VARCHAR(20), 

tel VARCHAR (15), 

remark VARCHAR(200), 

UNIQUE (name) 
» 


执行 上 面 的 语句 ， 就 可 以 为 水 果 信息 表 中 的 名 称 列 添加 唯一 约束 了 。 
4.6.2 在 修改 表 时 添加 唯一 约束 


在 创建 表 时 添加 唯一 约束 有 两 种 方法 ， 而 在 修改 表 时 添加 唯一 约束 只 有 1 种 方法 。 读 
者 可 以 对 比 之 前 学 习 过 的 几 种 约束 ， 看 看 在 修改 表 时 添加 唯一 约束 有 什么 变化 。 另 外 ， 还 
需要 读者 注意 的 问题 就 是 ， 在 已 经 存在 的 表 中 添加 唯一 约束 ， 要 保证 添加 唯一 约束 的 列 中 
存放 的 值 没有 重复 的 。 在 修改 表 时 添加 唯一 约束 的 语法 如 下 : 
ALTER TABLE table name 
ADD CONSTRAINT uq name UNIQUE (col name); 
其 中 : 
口 table name: 表 名 。 
口 CONSTRAINT uq_name: 添加 名 为 uq_name 的 约束 。 该 语句 可 以 省 略 ， 省 略 后 系 
统 会 为 添加 的 约束 自动 生成 一 个 名 字 。 
口 UNIQUE (col_name): 唯一 约束 的 定义 。UNIQUE 是 唯一 约束 的 关键 字 ，col_name 
是 表 中 的 列 名 。 如 果 想 要 同时 为 多 个 列 设置 唯一 约束 ， 就 要 省 略 掉 唯一 约束 的 名 
字 ， 名 字 由 系统 自动 生成 。 
现在 就 来 演练 一 下 上 面 的 语法 吧 ， 多 动手 才能 学 得 快 啊 ! 
【示例 26】 给 水 果 信 息 表 〈fruitinfo) 中 的 供应 商 联系 方式 〈tel) 加 上 唯一 约束 。 
将 水 果 信 息 表 中 的 供应 商 联系 方式 设置 成 唯一 约束 ， 语 名 如 下 : 
ALTER TABLE fruitinfo 
ADD CONSTRAINT uq fruitinfo tel UNIQUE (tel); -- 添 加 唯一 约束 


执行 上 面 的 语句 ， 就 为 水 果 信 息 表 中 的 tel 列 添加 了 一 个 名 为 uq_fruitinfo_tel 的 唯一 
约束 。 


4.6.3 ”去除 唯 一 约束 
任何 一 种 约束 都 是 可 以 删除 的 ， 删 除 唯一 约束 的 方法 也 很 简单 。 根 据 前 面 删除 约束 的 
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经 验 ， 读 者 应 该 清楚 只 要 咱 知道 了 约束 的 名 字 就 可 以 删 掉 ， 就 好 像 知 道 了 电话 号 码 就 可 以 
打 电 话 一 样 。 删 除 唯一 约束 的 语法 如 下 : 

ALTER TABLE table name 

DROP CONSTRAINT uq name; 

口 table name: 表 名 。 

口 uq_name: 唯一 约束 的 名 字 。 

读者 已 经 非常 了 解 这 个 语法 了 ， 实 际 应 用 一 下 就 可 以 更 好 地 掌握 了 。 

【示例 27】 删除 水 果 信息 表 (fruitinfo) 中 供应 商 联 系 方式 〈tel) 列 的 唯一 约束 。 

供应 商 联 系 方式 (tel) 列 的 唯一 约束 是 在 示例 27 中 添加 的 ， 名 字 是 uq_fruitinfo_tel。 
删除 该 约束 的 语句 如 下 : 

ALTER TABLE fruitinfo 

DROP CONSTRAINT uq fruitinfo tel; -删除 唯一 约束 


执行 上 面 的 语句 ， 就 可 以 将 名 为 uq_fruitinfo_tel 的 唯一 约束 删除 了 。 
4.6.4 使 用 企业 管理 器 轻松 使 用 唯一 约束 


在 SQL Server 数据 库 中 ,通常 会 把 唯一 约束 和 索引 放 在 一 起 操作 。 关 于 索引 的 定义 将 
在 本 书后 面 的 章节 中 讲解 。 由 于 唯一 约束 不 需要 设置 任何 表达 式 ， 因 此 它 在 企业 管理 器 中 
的 设置 也 是 非常 简单 的 。 下 面 就 让 我 们 一 起 感受 一 下 在 企业 管理 器 中 是 如 何 使 用 唯一 约束 
的 吧 ! 对 于 唯一 约束 的 操作 仍然 参考 示例 25 一 示例 27 的 应 用 ， 在 企业 管理 器 中 演示 如 何 
添加 以 及 删除 唯一 约束 的 操作 。 

【示例 28】 在 企业 管理 器 中 ， 给 水 果 信 息 表 (fruitinfo〉 中 的 水 果 名 称 Cname) 加 上 
唯一 约束 。 

在 本 例 中 综合 了 示例 25 和 示例 26 的 应 用 ， 无 论 是 在 创建 表 的 时 候 添 加 唯一 约束 ， 还 
是 在 修改 表 时 添加 唯一 约束 都 需要 在 表 的 设计 界面 中 完成 。 只 不 过 在 创建 表 时 添加 唯一 约 
束 ， 需 要 填 入 表 的 字段 信息 。 为 了 满足 读者 迫不及待 的 心情 ， 本 例 主 要 演示 如 何在 修改 该 
表 时 添加 唯一 约束 。 给 水 果 信 息 表 中 的 水 果 名 称 添加 唯一 约束 分 为 如 下 3 个 步骤 。 

(1) 打开 水 果 信息 表 的 设计 页 面 

在 企业 管理 器 的 对 象 资源 管理 器 中 ， 找 到 水 果 信息 表 ， 然 后 右 击 该 表 ， 在 弹出 的 右键 
菜单 中 选择 “设计 ”选项 ， 弹 出 图 4.35 所 示 界 面 。 

(2) 打开 添加 唯一 约束 的 界面 

要 给 表 添 加 唯一 约束 , 右 击 表 的 设计 页 面 , 在 弹出 的 右键 菜单 中 选择 “索引 / 键 ”选项 ， 
弹出 图 4.36 所 示 界 面 。 

从 图 4.36 的 界面 中 ， 可 以 看 到 在 没有 添加 唯一 约束 时 ， 表 中 就 已 经 有 了 一 条 信息 。 请 
读者 仔细 看 一 下 ， 不 难 发 现 这 个 信息 就 是 之 前 创建 的 主键 信息 。 也 就 是 说 在 这 个 界面 中 显 
示 的 信息 有 主键 信息 也 有 唯一 键 的 信息 ， 实 际 上 还 有 索引 的 信息 。 

(3) 添加 唯一 约束 

在 图 4.36 所 示 的 界面 中 ， 单 击 “ 添 加 ”按钮 ， 界 面 变 成 图 4.37 所 示 的 界面 。 
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图 4.37 添加 唯一 约束 界面 
既然 在 一 个 界面 中 可 以 添加 多 种 对 象 ， 就 需要 对 该 界面 进行 解读 了 ， 有 具体 说 明 如 下 。 


口 类 型 : 选择 要 添加 键 的 类 型 ， 分 为 两 种 一 一 索引 和 唯一 键 。 虽 然 在 显示 的 时 候 有 
主键 的 选项 , 但 是 主键 约束 不 是 在 这 个 界面 添加 的 , 请 参考 本 章 中 4.2 节 主 键 约束 
的 使 用 。 

口 列 : 选择 要 设置 成 唯一 约束 /索引 的 列 。 

口 是 唯一 的 : 有 两 个 选项 , 一 个 是 “是 ”一 个 是 “ 否 ”。 通常 唯一 约束 都 会 选择 “是 ”。 

口 名 称 : 唯一 约束 /索引 的 名 字 。 通 常 唯一 约束 的 名 字 以 UQ 开头 ， 索引 的 名 字 以 IX 
开头 。 

在 图 4.37 所 示 的 界面 中 , 为 水 果 信息 表 中 的 水 果 名 称 列 添加 唯一 约束 的 设置 如 图 4.38 

所 示 。 
在 图 4.38 的 界面 中 ， 单 击 “ 关 闭 ” 按 钮 ， 并 保存 表 的 信息 即 可 完成 对 水 果 信息 表 中 水 
果 名 称 列 唯一 约束 的 添加 操作 。 


外 说 明 : 细心 的 读者 会 发 现 ， 在 图 4.38 中 的 列 名 name 后 面 有 一 个 “ (ASC) ”。 它 是 什 
么 意思 呢 ? ASC 是 升序 排列 的 意思 ， 除 了 ASC 之 外 还 有 DESC， 也 就 是 降序 排 
列 的 意思 。 这 种 排序 主要 体现 在 数据 表 的 查询 中 ， 目 前 还 看 不 出 什么 效果 。 设 置 
是 升序 还 是 降序 ,是 在 选择 列 的 时 候 指定 的 ,选择 列 时 会 出 现 图 4.39 所 示 的 界面 。 


“Ws 


第 2 篇 ， 表 操作 基础 


ETIE E23| 
过 定 的 主 /唯一 键 或 索引 G); 


PR_fruitinfo 唯一 键 或 索 5| 的 属性 。 
岗 _fruitinfo_namet 


日 (号 规 ) a| 
类 型 唯一 键 
列 nane (ASC) 
日 标识 
FEE 
说 明 
日 表 设 i 


图 4.38 水 果 信 息 表 中 设置 水 果 名 称 为 唯一 约束 
可 世 


指定 用 于 此 索引 的 列 和 排序 顺序 C) 


列 名 排序 顺序 
到 升序 


图 4.39 ”选择 列 界面 
在 图 4.39 所 示 的 界面 中 ， 就 可 以 选择 列 名 和 排序 的 顺序 了 。 


【示例 29】 删除 水 果 信息 表 中 水 果 名 称 列 的 唯一 约束 。 

删除 约束 永远 都 是 相对 容易 的 事 ， 删 除 唯一 约束 也 不 例外 。 通 过 下 面 两 个 步 又 就 可 以 
轻松 地 将 唯一 约束 删除 了 。 

(1) 打开 水 果 信 息 表 的 设计 界面 

与 示例 28 的 第 1 个 步骤 一 样 ， 请 读者 自行 参照 该 步骤 打开 ， 这 里 就 不 再 多 说 了 。 

(2) 打开 索引 / 键 界 面 并 删除 唯一 约束 

打开 索引 / 键 界面 的 方法 与 示例 28 的 第 2 个 步骤 一 致 ， 打 开 后 的 界面 如 图 4.40 所 示 。 
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图 4.40 索引/ 键 界面 
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在 图 4.40 所 示 界 面 中 ， 选 择 UQ_fruitinfo_ name 的 唯一 约束 ， 单 击 “删除 ”按钮 ， 然 
后 再 单 击 “关闭 ”按钮 并 保存 表 的 信息 ， 即 可 完成 对 水 果 信 息 表 中 水 果 名 称 列 唯一 约束 的 
删除 操作 。 


4.7 非 空 约束 一 一 NOT NULL 


非 空 约束 就 是 用 来 确保 列 中 必须 要 输入 值 的 一 种 手段 。 有 时 设置 其 他 约束 时 ， 也 会 自 
动 将 该 列 设置 成 非 空 约束 的 ， 比 如 : 设置 主键 约束 时 就 会 将 该 列 自 动 设 置 成 非 空 约束 。 非 
空 约束 也 可 以 理解 成 是 检查 约束 的 一 种 ， 要 求 在 该 列 中 必须 输入 值 。 在 实际 的 应 用 中 ， 非 
空 约束 也 是 非常 必要 的 ， 比 如 : 在 网 上 注册 一 个 用 户 信息 时 ， 必 须要 输入 用 户 名 、 密 码 等 
必要 信息 ， 和 否则 注册 的 用 户 就 毫 无 意义 ， 同 时 也 会 给 数据 库 中 增加 很 多 垃圾 信息 。 


4.7.1 在 建 表 时 添加 非 空 约束 


非 空 约束 通常 都 是 在 创建 数据 表 时 就 添加 了 ， 添 加 非 空 约束 是 很 简单 的 。 添 加 的 语法 
也 就 一 种 ， 并 且 在 数据 表 中 也 可 以 为 同 列 设置 唯一 约束 。 当 然 ， 设 置 主键 约束 的 列 就 不 必 
再 设置 非 空 约束 了 。 下面 就 让 我 们 感受 一 下 最 简单 的 约束 是 如 何 添加 的 。 具体 的 语法 如 下 : 

CREATE TABLE NAME table name 

( 

COLUMN NAME1 DATATYPE NOT NULL, 


COLUMN NAME2 DATATYPE NOT NULL, 
COLUMN NAME3 DATATYPE 


读者 对 上 面 的 语法 已 经 不 陌生 了 吧 ， 没 错 ， 添 加 非 空 约束 就 是 在 列 的 数据 类 型 后 面 加 
上 NOT NULL 关键 字 。 上 面 的 语法 中 还 有 一 个 特点 ， 就 是 没有 给 唯一 约束 设置 名 字 ， 其 实 
唯一 约束 是 本 章 学 过 的 约束 中 唯一 一 个 没有 名 字 的 约束 。 
【示例 30】 根据 表 4-2 的 要 求 ， 创 建 水 果 信 息 表 ， 要 求 水 果 的 名 称 不 能 为 空 ， 价 格 也 
不 能 为 空 。 
创建 水 果 信 息 表 时 ， 要 将 水 果 信 息 表 中 的 水 果 编 号 设置 成 主键 ， 因 此 也 相当 于 不 能 为 
空 了 。 创 建 的 具体 语句 如 下 : 
CREATE TABLE fruitinfo 
. id INT PRIMARY KEY, 
name VARCHAR(20) NOT NULL, 
price DECIMAL(6,2) NOT NULL, 
origin VARCHAR(20), 
tel VARCHAR (15), 
remark VARCHAR(200) 
执行 上 面 的 语句 ， 就 可 以 完成 创建 水 果 信 息 表 并 为 水 果 信息 表 中 的 名 称 和 价格 列 加 上 
非 空 约束 了 。 运 行 效果 如 图 4.41 所 示 。 
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图 4.41 创建 水 果 信 息 表 并 添加 非 空 约束 


4.7.2 在 修改 表 时 添加 非 空 约束 


非 空 约束 比较 特殊 ， 与 前 面 所 讲 过 的 其 他 约束 在 添加 时 大 不 相同 。 请 读者 认真 学 习 它 
的 不 同 之 处 。 实 际 上 ， 在 修改 表 时 添加 非 空 约束 与 在 数据 表 中 修改 列 的 定义 是 相似 的 。 具 
体 的 语法 如 下 : 

ALTER TABLE table name 

ALTER COLUMN col name datatype NOT NULL; 

其 中 : 

口 table name: 表 名 。 

口 col_name: 列 名 。 要 为 其 加 上 非 空 约束 的 列 。 

口 datatype: 数据 类 型 。 列 的 数据 类 型 ， 如 果 不 修改 数据 类 型 ， 还 要 使 用 原来 的 数据 

类 型 。 

口 NOT NULL: 非 空 约束 的 关键 字 。 

通过 上 面 的 语法 就 可 以 很 容易 地 对 表 中 的 列 添加 唯一 约束 了 。 下 面 就 来 体会 一 下 它 的 
作用 的 吧 。 

【示例 31】 为 水 果 信 息 表 (fruitinfo) 中 的 供应 商 联系 方式 列 〈tel) 添加 非 空 约束 。 

在 添加 非 空 约束 之 前 ， 先 要 查看 一 下 水 果 信息 表 中 供应 商 联系 方式 列 的 数据 类 型 ， 然 
后 再 进行 添加 。 查 询 后 可 以 知道 ，tel 列 的 数据 类 型 是 VARCHAR(15)。 同 时 ， 如 果 在 水 果 
信息 表 中 的 供应 商 联系 方式 列 里 已 经 存在 了 空 的 记录 ， 那 么 需要 将 空 的 记录 删除 或 更 改 成 
其 他 信息 ， 否 则 无 法 添加 非 空 约束 。 添 加 非 空 约束 的 语句 如 下 : 

ALTER TABLE fruitinfo 

ALTER COLUMN tel VARCHAR(15) NOT NULL; 

执行 上 面 的 语句 后 ， 就 可 以 将 水 果 信 息 表 (fruitinfo) 中 的 供应 商 联系 方式 (tel) 设置 
成 非 空 约束 了 。 运 行 效果 如 图 4.42 所 示 。 
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EEC [iolxl 
文件 四 ”篇 辑 下 ) 视图 WD) 查询 @) 项 目 中 调式) 
IAD gOV EO 帮助 中 
罗 DD | 六 | 盖 名 名 DBS ss 
a 38 | chspter | 执行 Ch 加 
Leryl s... ator (55))* 
ALTER TABLE fruitinfo 
ALTER COLUMN tel VARCHAR(15) NOT NULL; 


bos C.. | vrow-9527\Aaninistrato... | chapter4 |00:00:00 |0 行 


就 绪 行 3 列 !1 chi 


图 4.42 给 水 果 信息 表 中 供应 商 联系 方式 列 添加 非 空 约束 


4.7.3 去 除非 空 约束 


非 空 约束 的 删除 与 其 他 约束 也 是 不 同 的 ， 由 于 非 空 约束 没有 名 字 ， 因 此 不 能 够 像 之 前 
学 习 的 删除 约束 的 方法 来 删除 .读者 可 以 思考 一 下 ,没有 设置 非 空 约束 的 列 用 什么 表示 呢 ? 
答案 是 用 NULL 表示 , 但 是 这 个 NULL 可 以 省 略 不 写 。 也 就 是 说 ， 某 个 列 要 取消 非 空 约束 
就 意味 着 该 列 可 以 为 空 , 因此 ,可 以 用 在 修改 时 添加 非 空 约束 的 语法 , 将 NOT NULL 变 成 
NULL 就 可 以 了 。 具 体 的 语法 如 下 : 


ALTER TABLE table name 
ALTER COLUMN col name datatype NULL; 


这 下 读者 高 兴 了 吧 , 不 用 再 多 记 一 个 新 语法 了 ， 只 要 将 上 一 小 节 语 法 中 的 NOT NULL 
换 成 NULL 就 可 以 了 。 下 面 读者 就 可 以 根据 上 面 的 语法 ， 将 示例 31 中 的 水 果 信 息 表 里 的 
供应 商 联系 方式 改 成 可 以 为 空 的 形式 。 


4.7.4 使 用 企业 管理 器 轻松 使 用 非 空 约束 


非 空 约束 用 企业 管理 器 来 管理 是 更 加 容易 的 ， 应 该 说 ， 非 空 约束 是 最 适合 在 企业 管理 
器 操作 的 了 。 在 企业 管理 器 中 操作 非 空 约束 ， 真 的 是 只 需 动 动 鼠 标 就 可 以 了 。 那 么 ， 现 在 
要 做 的 就 是 找到 动 鼠 标的 位 置 即 可 。 下 面 就 通过 一 个 综合 示例 来 讲解 如 何在 企业 管理 器 中 
管理 非 空 约束 。 

【示例 32】 使 用 示例 30 运行 后 的 水 果 信息 表 。 对 其 中 的 水 果 产 地 列 (origin) 添加 非 
空 约束 ， 取 消 水 果 名 称 列 (name) 的 非 空 约束 。 

一 部 分 读者 一 定 找到 了 非 空 约束 操作 时 动 鼠标 的 位 置 了 吧 。 如 果 还 没 找到 ， 现 在 就 告 
诉 你 ， 动 鼠标 的 位 置 就 在 水 果 信息 表 的 设计 页 面 中 。 在 本 例 中 添加 或 取消 非 空 约 束 ， 都 使 
用 下 面 的 两 个 步骤 就 可 以 完成 。 

(1) 打开 表 的 设计 界面 

在 企业 管理 器 的 对 象 资源 管理 器 中 ， 右 击 水 果 信息 表 ， 在 弹出 的 右键 菜单 中 选择 “ 设 
计 ” 选 项 ， 即 可 看 到 在 完成 了 示例 30 的 语句 后 水 果 信息 表 的 设计 界面 ， 如 图 4.43 所 示 。 

在 图 4.43 中 设置 非 空 约束 动 鼠 标的 位 置 ， 就 在 数据 类 型 后 面 的 “允许 Null 值 ”的 列 
中 。 从 图 上 可 以 看 出 id、name 和 price 列 的 “允许 Null 值 ” 列 都 处 于 未 选中 的 状态 ， 这 就 
表明 它们 是 设置 了 非 空 约束 的 。 相 反 ， 剩 余 列 的 “允许 Null 值 ” 列 都 处 于 选中 状态 ， 这 就 


ss 
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表明 它们 是 可 以 为 空 的 。 

(2) 根据 要 求 设置 非 空 约束 

按照 题目 的 要 求 ， 将 水 果 信 息 表 中 的 水 果 产 地 列 〈origin) 中 的 “允许 Null 值 ” 列 里 
的 选中 状态 去 掉 ; 将 水 果 名 称 列 Cname) 中 的 “允许 Null 值 ” 列 里 的 未 选中 状态 变 成 选中 
状态 。 设 置 后 ， 效 果 如 图 4.44 所 示 。 


文件 四 编辑 下 ) 视图 WW 项 目 F) 调试 加) 
素 设 计 器 (工具 加 窗口 WD 社区 C) 帮助 区 


卫 闽 建 查 放 中 | 记 | 也 也 四 | 巴 | 蕊 回忆 | 怠 下 


文件 到 ) ”编辑 有 视图 @) 项 目 EE) 调式 四 ) 
囊 设 计 器 Q) 工具 CD) 窗口 如 社区 帮助 0 


TE 
| 了 双 癌 郧 训 加 六 局 


| YIDY-9527\0. .. 0. fruitinfo| 


x 
光 评 Nl 值 | 
| 


图 4.43 水 果 信息 表 的 表 设计 界面 图 4.44 水 果 信息 表 中 设置 非 空 约束 的 效果 
在 完成 了 图 4.44 的 设置 后 ， 要 记得 保存 表 信息 。 这 样 ， 就 完成 了 在 水 果 信 息 表 中 的 非 
空 约束 设置 


在 示例 32 中 并 没有 讲解 在 创建 表 时 如 何 设置 非 空 约束 ， 其 实 是 与 修改 表 时 设置 非 空 
约束 一 样 的 ， 只 不 过 在 创建 表 时 还 需要 将 表 的 字段 信息 一 起 加 入 。 


全 注意 ; 在 企业 管理 器 中 操作 表 时 ， 保 存 表 数据 时 经 常会 出 现 如 图 4.45 所 示 的 界面 。 


图 4.45 ”保存 表 时 出 现 的 错误 


在 这 个 错误 提示 中 ， 可 以 看 到 并 不 是 修改 表 的 错误 而 是 对 表 的 设置 阻碍 了 修改 数 
据 表 。 要 更 改 这 个 设置 ， 只 需要 单 击 企业 管理 器 中 菜单 栏 的 “工具 ”|“ 选 项 ” 命 
令 ， 在 弹出 的 选项 界面 中 ， 展 开 Designers 选项 ， 并 选择 “ 表 设 计 器 和 数据 库 设 
计 器 ”选项 ， 如 图 4.46 所 示 。 
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EE Ed] 
[而 
2 设计 器 更 新 的 连接 字符 
中 和 Ri 审 超时 值 中) 
由 查询 结果 
让 SQL Server 对 外商 源 和 理 器 超过 此 时 间 后 事务 超时 GE): 
Das Fo “ 呀 


维护 计划 
由 -Analysis Services 设计 器 厂 出 现 full 主键 时 警 省 外) 
四 访 代 亲生 理 


克 表 妥 到 影响 时 警 省 CT) 
| 阻止 保存 要 求 重新 他 际 表 的 更 改 
EE 加 本 了 


碟 在 新 建 关系 图 时 启动 “添加 表 ” 对 话 框 L) 


大- 列 朵 .本 | 
图 4.46 “工具 ”菜单 下 的 “选项 ”界面 


在 图 4.46 所 示 的 界面 中 ， 可 以 看 到 “阻止 保存 要 求 重新 创建 表 的 更 改 ” 选 项 处 于 
选中 状态 。 去 掉 该 选项 的 选中 状态 后 ， 即 可 完成 数据 表 的 保存 操作 。 


4.8 本 章 小 结 


在 本 章 中 主要 讲解 了 SQL Server 数据 库 中 的 6 种 约束 : 主键 约束 、 外 键 约束 、 默 认 值 
约束 、 检 查 约束 、 唯 一 约束 以 及 非 空 约束 的 使 用 方法 。 通 过 对 这 6 种 约束 的 使 用 方法 的 讲 
解 ， 读 者 应 该 能 够 熟练 地 掌握 使 用 T-SQL 如 何 操作 约束 ， 以 及 在 企业 管理 器 中 如 何 操作 约 
束 。 另 外 ， 通 过 本 章 的 学 习 ， 读 者 也 应 该 能 够 找到 使 用 工 SQL 语句 操作 这 6 种 约束 时 语法 
的 一 些 规律 ， 找 到 了 这 些 规律 就 能 方便 读者 对 语法 的 理解 和 记忆 。 只 要 读者 在 数据 表 中 灵 
活 使 用 这 些 约束 ， 一 定 会 使 数据 表 中 数据 的 完整 性 有 所 加 强 的 ! 


4.9 本 章 


一 、 填 空 题 


1. 约束 主要 有 
2. 外 键 约束 的 关键 字 是 
3. 默认 值 约束 的 关键 字 是 


二 、 选 择 题 
1. 关于 主键 约束 描述 正确 的 是 


A. 一 张 表 中 可 以 有 多 个 主键 约束 B. 一 张 表 中 只 能 有 一 个 主键 约束 
C. 主键 约束 只 能 由 一 个 字段 组 成 D. 以 上 都 不 对 
2. 下 面 哪 一 个 约束 要 涉及 两 张 表 
A. 外 键 约束 B. 主键 约束 
C. 检查 约束 D. 唯一 约束 
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3. 下 面 对 检 查 约束 描述 正确 的 是 
A. 一 个 列 可 以 设置 多 个 检查 约束 B. 一 个 列 只 能 设置 一 个 检查 约束 
C. 检查 约束 中 只 能 写 一 个 检查 条 件 D. 以 上 都 不 对 

三 、 问 答题 


1. 约束 的 作用 是 什么 ? 
2. 为 什么 要 使 用 默认 值 约束 ? 
3. 主键 约束 和 唯一 约束 的 区 别 是 什么 ? 


、 操 作 题 


使 用 表 4-2 所 示 的 水 果 信息 表 (fruitinfo〉 完 成 如 下 的 约束 操作 。 
(1) 给 水 果 信 息 表 中 的 编号 列 设置 主键 约束 。 

(2) 给 水 果 价 格 列 设置 检查 约束 ， 要 求 水 果 价 格 在 1 一 10 元 之 间 。 
(3) 给 水 果 名 称 列 设置 唯一 约束 。 

(4) 删除 之 前 设置 的 所 有 约束 。 
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通过 前 面 几 章 的 学 习 ， 读 者 已 经 清楚 如 何 创 建 数据 库 和 数据 表 了 。 但 是 ， 只 有 数据 库 
和 数据 表 是 没有 什么 用 的 ， 这 就 好 像 是 一 个 清水 房 什么 都 没有 是 无 法 居住 的 。 如 果 想 要 住 
在 这 个 房子 里 ， 就 需要 对 这 个 清水 房 装修 。 装 修 就 是 在 清水 房 中 使 用 一 些 装饰 材料 以 及 一 
些 家 居 用 品 等 。 数 据 库 也 是 需要 装修 才能 够 使 用 的 ! 那么 ， 在 数据 库 中 使 用 的 装修 材料 是 
什么 呢 ? 呵呵 ， 只 能 是 数据 了 鹃 。 如 何 用 数据 装修 数据 表 呢 ? 请 读者 认真 学 习 本 章 带 给 你 
的 内 容 吧 。 

本 章 的 主要 知识 点 如 下 : 

口 如 何 向 数据 表 中 添加 数据 

口 修改 数据 表 中 的 数据 

口 删除 数据 表 中 的 数据 


5.1 向 数据 表 中 添加 数据 一 -INSERT 


装修 房子 的 第 一 步 就 是 要 将 购买 的 材料 放 到 清水 房 中 ， 有 了 这 些 材料 就 能 够 开始 装修 
房子 了 ， 否 则 只 能 看 着 清水 房 了 。 同 样 的 道理 ， 数 据 表 中 有 了 数据 ， 数 据 库 才 有 意义 啊 ! 
那么 ， 如 何 向 数据 表 中 添加 数据 ， 就 是 管理 数据 表 的 一 个 重要 环节 了 。 向 数据 表 中 添加 数 
据 可 以 通过 SQL 语句 ， 也 可 以 通过 使 用 企业 管理 器 来 完成 。 下 面 就 给 读者 一 一 道 来 吧 ! 


5.1.1 INSERT 语句 的 基本 语法 形式 


使 用 SQL 语句 向 数据 表 中 添加 数据 前 , 要 先 弄 清 楚 添 加 语句 的 语法 规则 。 读者 先 要 记 
住 的 是 添加 语句 使 用 的 关键 字 INSERT。 如 果 以 后 你 看 到 INSERT 开头 的 SQL 语句 ， 就 可 
以 想 想 这 是 不 是 向 数据 表 添 加 数据 的 操作 呢 。 在 实际 应 用 中 添加 语句 的 语法 形式 有 很 多 ， 
在 本 小 节 中 介绍 一 个 比较 常用 的 INSERT 语句 的 语法 形式 ， 具 体格 式 如 下 所 示 。 
INSERT INTO table name(column namel, column name2,......) 
VALUES (valuel,value2,......); 
其 中 : 
口 table name: 表 名 。 在 向 该 表 中 添加 数据 时 不 要 忘 了 要 将 表 所 在 的 数据 库 打开 。 
口 column_name: 列 名 。 需 要 添加 数据 的 列 名 ， 如 果 没 有 指定 任何 列 名 ， 意 味 着 是 向 
表 中 所 有 列 添加 数据 。 
口 value: 值 。 向 数据 表 中 指定 列 添加 的 值 ， 值 与 表 中 的 列 是 一 一 对 应 的 。 不 仅 个 数 
要 一 致 ， 数 据 类 型 也 要 一 致 才 可 以 。 另 外 ， 如 果 没 有 指定 列 名 ， 插 入 值 对 应 列 的 
顺序 就 是 数据 表 中 列 的 存放 顺序 。 
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5.1.2 ”给 表 里 的 全 部 字段 添加 值 


向 表 中 所 有 的 字段 同时 插入 值 ， 是 
添加 值 也 是 INSERT 语句 形式 中 最 简 身 


一 个 比较 常见 的 应 用 。 实 际 上 ， 向 表 中 的 全 部 字段 
的 一 种 了 。 下 面 就 用 示例 1 来 演示 一 下 如 何 应 用 


INSERT 语句 向 表 中 全 部 字段 添加 值 。 
表 吧 ， 如 表 5-1 所 示 。 


在 演示 之 前 ， 先 来 了 解 一 下 在 本 章 中 要 操作 的 数据 


表 5-1 游戏 账号 信息 表 (gameaccount) 
编号 列 名 中 文 释义 
1 id 账号 
2 name VARCHAR(20) 昵称 
3 password VARCHAR(20) 密码 
4 level INT 等 级 
5 gamegroup VARCHAR(20) 游戏 组 
6 points 积分 
7 email 邮箱 
8 remark 备注 


根据 表 5-1 的 结构 ， 创 建 游戏 账号 信息 表 〈gameaccount) 的 语句 如 下 所 示 。 另 外 ， 本 


章 的 数据 表 全 部 创建 在 数据 库 chapter5 


USE chapter5; 
CREATE TABLE gameaccount 
( 
id INT PRIMARY KEY, 
name VARCHAR(20), 
password VARCHAR(20), 
level INT, 
gamegroup VARCHAR(20), 
points INT, 
email VARCHAR(20), 
remark VARCHAR(200) 
); 


中 。 


【示例 1】 向 游戏 账号 信息 表 〈gameaccount〉 中 添加 数据 ， 如 表 5-2 所 示 。 
表 5-2 游戏 账号 信息 表 添 加 的 数据 
账号 密码 等 级 游戏 组 邮箱 备注 
1 112233 1 初级 联盟 pao@126.com 无 


向 游戏 账号 信息 表 (gameaccount) 


USE chapter5; 
INSERT INTO gameaccount 


中 添加 表 5-2 所 示 的 数据 ， 具 体 的 语句 如 下 : 


VALUES (1, ' 跑 得 快 ', '112233' ,1, ' 初 级 联盟 ',100, 'pao@126.com', ' 无 '); 


执行 上 面 的 语句 ， 就 可 以 向 游戏 账号 信息 表 中 添加 一 条 数据 了 。 执 行 效果 如 图 5.1 


所 示 。 
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也 新建 查询 | 出 | 孔 记 马扎 | 区 加 局 | 怠 司 
-| 执行 》 和 江村 国 | 六 桓 | 汐 央 克 5 


由 INSERT INTO gameaccount 


VALUES | 1, ' 跑 得 快 ',' 112233' ,1,' 初 级 联盟 ' ,100,' pao8126.com',' 无 '); 局 


况 查 … | tID#-9s27WhssqLsERYER2008 (.. | WID-9527\Adninistrato... | chspter5 |00:00:00 |0 行 
就 绪 行 5 列 1 Chl Je /4 


图 5.1 向 游戏 账号 信息 表 (gameaccount) 中 全 部 字段 插入 数据 
从 图 5.1 中 ， 可 以 看 到 执行 上 面 的 语句 后 给 出 的 消息 是 “1 行 受 影响 ” 这 就 意味 着 有 
一 条 数据 插入 到 数据 表 中 了 。 读 者 是 不 是 很 想 查 看 一 下 数据 表 中 是 否 有 这 条 数据 呢 ? 很 简 
单 ， 记 住 下 面 的 语句 就 可 以 查看 到 表 中 的 数据 了 。 
SELECT * FROM table name; 
其 中 ，table_name 就 是 表 的 名 字 。 这 里 ， 将 table_name 换 成 是 gameaccount 就 可 以 查 
询 了 。 查 询 效果 如 图 5.2 所 示 。 


R、microsoft SQL Server Wanagenent Stadio =Iolx! 
文件 中 ”编辑 到 ) 视图 查询 @) 项 目 人 E) 调式) 工具 YD) 窗口 中) 
社区 @C 帮助 0 
也 新建 查询 | 书 | 沁 六 名 | 向 | 访 回 马 | 加 局 
a | chapters ~ ?执行 Oh 有 v 弹 本 回 | 3 总 
SQLQueryl. s... ator (52))*| vx 

日 USE chapter5 ~ 


SELECT * FROM gameaccount; 


-9527\MSSQLSERVER2008 〔 . ，| WIDW-9527\Adninistrato. . | chapter5 |00:00:00 |1 行 
就 绪 行 4 列 2 Ch2 用 


图 5.2 ”游戏 账号 信息 表 (gameaccount) 中 的 数据 
看 到 了 图 5.2 所 示 的 查询 结果 ， 读 者 这 回放 心 了 吧 ， 数 据 真 的 存放 在 数据 表 中 了 。 


5.1.3 ”给 需要 的 字段 添加 值 


在 实际 工作 中 ， 添 加 数据 时 并 不 是 每 次 都 需要 向 表 中 的 所 有 字段 添加 值 的。 当然 ， 这 
个 问题 也 不 难 解决 ， 还 是 用 INSERT 语句 就 可 以 解决 了 ， 只 不 过 在 插入 数据 时 需要 指定 字段 
名 才 可 以 。 学 习 任 何 SQL 语句 ， 实 际 操作 是 最 重要 的 了 。 下 面 就 让 我 们 一 起 走 进 示例 2 吧 。 
【示例 2】 向 游戏 账号 信息 表 (gameaccout) 中 插入 表 5-3 所 示 的 数据 。 


表 5-3 ”游戏 账号 信息 表 添 加 的 数据 


账号 昵称 密码 ”| 等 级 游戏 组 积分 邮箱 备注 
2 乐 呵呵 123456 | 2 中 级 联盟 230 


者 
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从 表 5-3 中 可 以 看 出 邮箱 和 备注 两 个 字段 是 不 需要 添加 值 的 ， 因 此 ， 添 加 的 SQL 语句 
如 下 : 


USE chapter5; 


INSERT INTO gameaccount (id,name,password,1level,gamegroup,points) 
VALUES (2, ' 乐 呵呵 ', '123456' ,2, ' 中 级 联盟 ' ,230); 


执行 上 面 的 语句 ， 就 可 以 向 游戏 账号 信息 表 (gameaccount) 中 填 入 一 条 数据 了 。 执 行 
效果 如 图 5.3 所 示 。 


rosoft SQL Server Wanagenent Studio 
文件 中 编辑 EE) 视图 四 查询 @) 项 目 @) 调式 0) 工具 四 窗口 中 社区 CC 帮助 吕 
9 | 访 | 填 也 久 | 访 | 罗 回电 | 网 局 

MY Wa | chepters -| 执行 Wy 》 有 v 品 晤 国 | 站 列 | 的 疾 妈 | 三 2 名 


SQLQuery2. s... ator (55))#| vx 
USE chapters; 


INSERT INTO gameaccount (id, name,password, level, gamegroup, points) 


Pa 2,' 乐 呵呵 ,,'123456' ,2,' 中 级 联盟 , ,230) ; 
4 


加 查询 已 成 功 … | WID-9527 WRSSQLSERYER2008 〔 . ，| WIDW-9527\Adninistrato..，| chapter5 |00:00:00 |0 行 


就 绪 行 2 列 BT Ch 67 Ins 5 
图 5.3 向 游戏 账号 信息 表 指定 字段 插入 值 
下 面 利用 SELECT 语句 查看 一 下 插入 数据 后 的 效果 吧 ， 如 图 5.4 所 示 。 


SQL Server Nanagenent Studio 玉 

文件 EF) 编辑 E) 视图 @) 查询 @) 项 目 @) 调试 @) 工具 C) 窗口 和 ) 社区 CC) 帮助 了 D 
卫 亲 寻 查询 四 | 出 | 也 也 怠 | 记 | 蕊 加 己 | 进 下 

RY WR | chapters |? 执行 DO 有 v 开本 避 3 哆 | 的 阅 | 如 | 
SQLquery2. s... ator (55))*| 
日 USE chapters; 
| SELECT * FRON gameaccount; 


初 弛 联盟 100 pao@126com 无 
| 加 是 2 乐器 可 123456 2 ”中 骸 联 盟 ”230 NULL NULL 


团 查 询 已 成 功 … | WIDt-9527\MISSQLSERYER2008 ( . ，| WIDW-9527\Adninistrato..、| chapter5 |00:00:00 | 2 行 
就 绪 行 2 列 31 Ch 31 Ins / 


图 5.4 游戏 账号 信息 表 (gameaccount) 中 的 数据 


从 图 5.4 查询 的 信息 中 ， 可 以 看 出 在 示例 2 中 没有 添加 的 数据 的 email 和 remark 列 的 
值 都 是 NULL 而 不 是 “”。 这 个 问题 需要 读者 注意 哦 ! 


5.1.4 给 自 增长 字段 添加 值 


什么 是 自 增 长 字段 呢 ? 就 是 在 第 3 章 中 给 读者 介绍 的 标识 列 ， 想 起 来 了 吧 。 自 增长 字 

段 只 能 在 整 型 的 字段 上 设置 ,并且 在 设置 时 可 以 为 其 指定 该 列 开始 的 值 以 及 每 次 增长 的 值 。 
读者 可 能 会 问 了 ， 自 增长 字段 都 已 经 有 值 了 ， 为 什么 还 需要 添加 值 呢 ? 另外， 直接 向 自 增 
长 字段 中 添加 值 可 以 吗 ? 既然 有 这 么 多 的 疑问 ， 现 在 就 给 读者 一 一 解答 吧 。 先 说 一 下 ， 为 
什么 会 有 向 自 增长 字段 添加 值 的 需求 ， 实 际 上 大 部 分 的 原因 是 由 于 自 增长 字段 的 值 只 能 按 


写生 
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顺序 增加 ， 如 果 将 其 中 的 某 一 个 序号 所 对 应 的 数据 删除 掉 ， 系 统 是 不 会 为 其 补充 该 值 的 。 
还 是 做 个 形象 点 的 解释 吧 ， 如 图 5.5 所 示 。 图 5.5 中 左边 的 表格 中 是 数据 表 的 全 部 数据 ， 
图 5.5 中 右边 的 表格 中 是 在 左边 表格 的 基础 上 去 掉 一 条 数据 后 的 效果 。 其 中 ， 账 号 列 是 自 
增长 字段 。 


和 
要 


密码 


账 

1 天 下 第 一 “| 112233 账号 二 昵称 宇 码 
2 最 牛人 223344 1 | 天 下 第 一 “| 112233 
3 就 受 沈 334455 | 一 一 > | 2 | 最 牛人 223344 
4 天 下 匹敌 445566 4 | 天 下 无 敌 。 | 445566 
5 青 定 高 556677 5 | 衣 定 高 556677 


图 5.5 数据 表 中 数据 删除 前 和 删除 后 的 效果 


从 图 5.4 中 ， 可 以 看 出 删除 一 条 数据 后 ， 账 号 所 在 列 值 就 缺少 了 3 这 个 值 。 如 果 继 续 
向 该 表 中 添加 数据 ， 账 号 列 也 不 会 再 添加 3 这 个 值 了 ， 而 是 继续 从 5 开始 向 上 加 。 

有 的 时 候 需 要 将 不 连续 的 值 补充 ,就 需要 先 将 自 增长 字段 的 自动 插入 属性 设置 成 OFF。 
具体 的 设置 语法 如 下 所 示 。 


SET IDENTITY INSERT table_name ON; 


这 里 ，IDENTITY_INSERT 就 是 自 增长 字段 自动 插入 值 的 属性 ，table_name 是 表 名 。 
如 果 要 将 该 属性 恢复 就 将 上 面 语句 中 的 ON 改 成 OFF 就 可 以 了 。 下面 就 用 示例 3 来 尝试 一 
下 吧 。 

【示例 3】 向 游戏 账号 信息 表 (gameaccountl ) 中 根据 表 5-4 添加 数据 。 

为 了 配合 自 增长 字段 的 使 用 ， 新 创建 一 个 游戏 账号 信息 表 〈gameaccount1)， 其 中 列 只 
取 gameaccount 表 中 的 前 3 列 〈 如 表 5-1 所 示 )， 并 将 其 中 的 账号 列 设置 成 自 增长 的 。 创 建 
gameaccountl 表 的 语法 如 下 : 

USE chapter5; 

CREATE TABLE gameaccountl1 

( 

id INT PRIMARY KEY IDENTITY(1,1), 
name VARCHAR(20), 


password VARCHAR(20), 
); 


通过 上 面 的 语句 ， 就 可 以 在 数据 库 chapter5 中 创建 数据 表 gameaccountl 了 。 


有 了 游戏 账号 信息 表 (gameaccount1) ， 就 可 以 向 该 表 插入 数据 了 。 要 添加 的 数据 如 
表 5-4 所 示 。 


表 5-4 需要 添加 的 数据 


密 码 
123456 
123123 


从 表 5-4， 可 以 看 出 需要 向 数据 表 中 添加 2 条 记录 ， 第 1 条 记录 中 的 账号 不 需要 输入 ， 
由 自 增长 字段 自动 添加 ; 第 2 条 记录 的 账号 列 手动 添加 。 具 体 的 添加 语句 如 下 : 


use chapter5; 
INSERT INTO gameaccountl1 


ss 
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VALUES (' 不 怕 输 '，'123456') 7 

SET IDENTITY INSERT gameaccountl1 ON; 
INSERT INTO gameaccountl (id,name,password) 
VALUES (3, ' 不 着 急 '，,'123123'); 

SET IDENTITY INSERT gameaccount1 OFF; 


执行 上 面 的 语句 ， 就 可 以 向 游戏 账号 信息 表 (gameaccount1〉 中 添加 2 条 数据 。 执 行 
效果 如 图 5.6 所 示 。 


向 游戏 账户 信息 表 (gameaccount1) 中 插入 2 条 记录 后 ， 查 询 游 戏 账户 信息 表 
(gameaccount1)， 效 果 如 图 5.7 所 示 。 


ET 本 中] 
文件 @) 编辑 和 视图) 查询 @) 项 目 调式 W) 工具 WD 窗口 四 
社区 CC 帮助 中 


:了 .有 建 查 询 中 | 由 也 也 四 | 包 | 蕊 四 雪 | 过 十 
*| ?3 行 O Ra v 如 富国 


文件 中。 编辑) 视 目 查询 @) 项 目 吕 ) 
调式 WW) 工具 中 窗口 如 ”社区 C) 帮助 吕 


m8 | 六 | 翅 也 本 | 忆 | 区 加 当 


VaLUES(' 不 怕 
SET IDENTITY 


[这 INTO g 
许 INTO 9 


VALUES (3,' 不 : 
SET IDENTITY_INSERT gan 


MY 2 | chapters ”| 和 执行 &) 总 


SqlQueryl. s... ator (53))* sx 
站 selecc * from gemeaccountl; 一 


人 1 行 受 影响 ) 
(1 行 受 影响 ) 了 下 
加 全 3 不 着 急 ”123123 
TVISSQLSERYER2008 〔 . ，| WIDN-9527\Adninistrato. .. | chspter5 |00:00:00 |0 行 [mw-ss27\WAanini strate ,，| chspter5 |00:00:00 |2 行 
就 请 行 了 列 区 Ch38 4 第 1 第 1 列 
图 5.6 向 自 增长 字段 插入 值 图 5.7 插入 2 条 记录 后 的 游戏 账户 信息 表 


从 图 5.7 的 添加 结果 可 以 看 出 ， 第 1 条 记录 中 id 列 使 用 了 自 增长 序列 的 第 1 个 值 ， 第 
2 条 记录 中 id 列 使 用 的 是 手动 添加 的 值 “3”。 问 读者 一 个 小 问题 吧 ， 如 果 再 向 游戏 账户 信 
息 表 (gameaccount1) 中 插入 值 时 ，id 列 中 的 值 是 “2” 还 是 “4” 呢 ? 记 住 , 一定 是 “4”， 
因为 自 增长 序列 一 定 是 在 该 列 的 最 大 值 基础 上 进行 增加 的 。 

很 多 读者 会 有 这 样 的 想法 吧 ,， 如 果 向 自 增长 字段 中 添加 值 ， 而 不 将 该 表 的 IDENTITY _ 
INSERT 属性 值 设 置 成 ON 会 出 现 什么 错误 呢 ? 下 面 仍然 以 向 游戏 账号 信息 表 
(Cgameaccount1) 插入 值 为 例 来 测试 如 下 的 语句 。 

USE chapter5; 

INSERT INTO gameaccountl1 

VALUES (2,' 游 戏 王 ' ,123321); 

执行 上 面 的 语句 会 发 生 什 么 呢 ? 会 发 生 如 下 的 错误 ， 如 图 5.8 所 示 。 

这 回 看 到 图 5.8 所 示 的 错误 提示 信息 了 吧 ，IDENTITY INSERT 属性 的 值 为 ON 时 才 
可 以 向 自 增长 字段 中 插入 值 。 除 了 这 个 提示 , 读者 还 会 发 现 一 个 信息 “ 仅 当 使 用 了 列 列表 ”， 
这 个 信息 的 意思 是 使 用 INSERT 语句 时 ， 要 加 上 插入 列 的 列 名 ， 否 则 也 无 法 向 自 增长 列 中 
插入 值 。 请 读者 根据 示例 3 的 语句 ， 自 己 尝 试 着 改 一 下 吧 ! 


5.1.5 向 表 中 添加 数据 时 使 用 默认 值 
在 第 4 章 中 介绍 约束 时 提 到 过 默认 值 约束 的 概念 ， 它 的 作用 是 设置 默认 值 约束 的 列 不 


.114 。 
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证 口 WD ”社区 局 和 助 吕 


EE TNE 


因 朗 | cpters -1#5w bs v 汇 写 国 | 玉 对 | 双 轿 和 | 三 全 | 未 训 芒 
‘SQLQuery! ato 一 


上 usF cha 
白 INsERT 


卜 查询 已 守成， 但 有 措 -…， | FID#-9S27 WSSQLSERYER2008 (TIN-952T7WAininistrato | chapters |00;00:00 |0 行 
职 境 行 3 列 本 三 4 


图 5.8 向 自 增长 列 插入 值 时 出 现 的 错误 


插入 值 时 自动 采用 已 经 设置 好 的 默认 值 。 比 如 : 游戏 账号 信息 表 (gameaccount) 中 的 备注 
列 ， 在 不 添加 具体 的 备注 信息 时 ， 默 认 让 其 添加 “无 ”。 这 种 情况 下 ， 可 以 在 创建 游戏 账号 
信息 表 (gameaccount) 时 ， 为 其 备注 字段 设置 默认 值 约 束 ， 默 认 值 是 “无 ”。 在 本 章 已 经 
创建 过 游戏 账号 信息 表 〈gameaccount) 了 ， 请 读者 使 用 下 面 的 语句 给 该 表 的 备注 列 加 上 默 
认 值 “无 ”。 

USE chapter5 


ALTER TABLE gameaccount 
ADD CONSTRAINT df gameaccount DEFAULT ' 无 ' FOR remark; 


执行 上 面 的 语句 后 ， 就 可 以 为 游戏 账号 信息 表 (gameaccount) 的 备注 字段 添加 默认 值 
约束 了 。 下 面 就 通过 示例 4 来 演示 默认 值 字段 的 使 用 效果 。 
【示例 4】 向 游戏 账号 信息 表 〈gameaccount〉 中 添加 如 表 5-5 所 示 的 数据 。 


表 5-5 ”游戏 账号 信息 表 添 加 的 数据 
| 也 称 | 密码 | 等 级 | 游 开 组 | 积分 | 邮箱 | 备注 
3 | 人 快 快 由 | 987654 | 1 | 初级 联盟 | 100 | paopao@126com | 无 
4 | 不 全 玩 | 123456 | 0 | 初级 联 明 | 10 | gamel21@126com | 积分 不 中 


根据 表 5-5 所 示 的 数据 ， 第 1 条 数据 中 的 备注 字段 是 “无 ” 可 以 在 插入 的 时 候 利 用 默 
认 值 。 具 体 的 语句 如 下 : 

USE chapter57 

INSERT INTO gameaccount (id,name,password,level,gamegroup,points, email) 

VALUES (3，' 快 快 跑 '，'987654'，1，' 初 级 联盟 '，,100, 'paopao@126.com'); 

INSERT INTO gameaccount 

VALUES (4, ' 不 会 玩 ', '123456' ,0,' 初 级 联盟 ' ,10, ' game121@126.com',' 积 分 不 足 '); 


执行 上 面 的 语句 ， 就 可 以 为 游戏 账户 信息 表 (gameaccount) 添加 2 条 数据 了 。 执 行 效 
果 如 图 5.9 所 示 。 

从 图 5.9 所 示 的 效果 ， 可 以 看 出 数据 已 经 加 入 到 账户 信息 表 中 ， 那 么 ， 第 1 条 数据 中 
备注 字段 的 值 究竟 是 不 是 “无 ” 呢 ? 下 面 就 来 查询 一 下 游戏 账户 信息 表 看 看 效果 , 如 图 5.10 
所 示 。 

哈哈 ， 在 图 5.10 中 id 列 是 3 的 这 条 记录 中 ，remark 列 的 值 是 “无 ” 吧 。 这 就 是 默认 
值 约束 的 应 用 哮 ! 


所 了 和 
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项 目 E) 调试 gj) 工具 GD) 
卫 新建 查询 思 | 由 也 也 怠 | 应 | 七 加 本 | 怠 瑟 

My ha | aeptes | 所 行 O 》 av 品名 国 | 下 马 | 痊 闫 瓜 | 三 全 | 地 
“SQLQueryl. s... ator (53))*| WIIN-S527\N meaccountIi# | -x 
章 USE chapcer5; 


INSERT INTO gameaccount (id, name,password, level,gamegroup,points,email) 
VALUES (3, ' 快 快 跑 '，' 987654' ,1,' 初 级 联盟 ' ,100, ' paopao8126.com'); 


窗口 轨 社区 加 帮助 只 


| YTDW-9527\SSQLSERYER2008 〔 -| WIDN-9527\Adninistrato. .| chapter5 |00:00:00 | 0 行 
0 列 1 ch 1 Ims 


文件 四 编辑 加、 视图 WW 查询 @) 项 目 @) 调 坛 ) 工具 窗口 WD 社 区 @) 
才 助 QD 


了 | 昼 | 古 号 久 | 刘 | 罗 日 马 | 如 
Ry Wa | chepters | ?执行 O 》 和 只 各 国 玉 哆 | 的 名 


SqlQueryl. s. .ator (53))*| WI newccontls | -x 
select * from gameaccount; 


NULL NULL 
paopao@1.， 无 
| | 4 不 会 玩 123456 game121.， 积分 不 足 


ID-9527\MSSQLSERVER2008 〔 .，| ITDW-952T\Adninistrate ,| ehspter5 |00:00:00 | 4 行 
就 结 第 ! 行 第 1 列 BA 


图 5.10 “游戏 账户 信息 表 (gameaccount) 中 的 数据 


5.1.6 ” 表 中 的 数据 也 能 复制 


如 果 你 


j 过 计算 机 ， 相 信 一 定做 过 复制 和 粘贴 的 操作 吧 ? 也 就 是 说 ， 在 计算 机 中 很 多 


的 文档 都 是 可 以 复制 的 ， 那 么 数据 表 中 的 数据 可 以 复制 吗 ? 当然 可 以 复制 了 ， 但 是 要 有 条 
件 限制 的 。 表 和 表 之 间 的 数据 复制 ， 要 遵守 列 的 个 数 和 数据 类 型 一 致 性 的 原则 。 并 且 ， 可 
以 有 选择 地 复制 表 中 一 部 分 列 中 的 数据 。 将 table_name2 中 的 数据 加 入 到 table_ namel 中 具 


体 的 语法 如 下 : 
INSERT INTO table namel (column namel, column name2,......) 
SELECT column name 1, column name 2,..... 


FROM table name2 


其 中 : 


table namel: 插入 数据 的 表 。 


column_name_1: table name2 中 的 列 名 。 


| 
口 column_ namel: 表 中 要 插入 值 的 列 名 。 
口 
口 


table name2: 取 数 据 的 表 。 


» Me 
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在 示例 学 习 之 前 ， 还 要 提醒 读者 在 INSERT INTO 后 的 列 名 个 数 与 SELECT 后 的 列 名 
个 数 要 一 致 ， 它 们 的 数据 类 型 也 要 一 致 。 下 面 就 通过 示例 $ 来 演示 如 何 复 制 表 中 的 数据 。 

【示例 S】 将 表 gameaccount 中 的 游戏 昵称 和 密码 复制 到 表 gameaccountl 中 。 
由 于 在 表 gameaccountl 中 编号 列 id 是 自 增 长 的 ， 因 此 ， 不 需要 再 向 该 列 赋值 了 。 根 
据 题目 要 求 ， 具 体 的 语句 如 下 : 

USE chapter5 

INSERT INTO gameaccount] (name,password) 


SELECT name,password 
FROM gameaccount; 


执行 上 面 的 语句 ， 就 可 以 将 gameaccount 表 中 的 数据 复制 到 gameaccountl 表 中 了 。 执 
行 效果 如 图 5.11 所 示 。 


从 图 5.11 中 可 以 看 出 ， 已 经 给 gameaccountl 中 添加 了 4 条 记录 。 查 询 gameaccountl 
表 效果 如 图 5.12 所 示 。 


microsoft SQL Server Nanagement Stuaii -Io|x| 
文件 四 编辑 于 ) 视图) 查询 @) 项目 中 调和 二 中) 

工具 中 窗口 WD 社区 忆 各 中 

卫 新 建 查询 记 | 这 也 怠 | 记 | 启 加 本 加 吾 
轩 巾 | htes "| > a 
SQLQueryl. s... ator (53))* x 
USE chapcer5 

SELECT * FROM gameaccount1; 


Raicrosoft SQL Server Nansgeseatl Stadio 2 =Iolx| 

文件 里 ) 编辑 是) 视图 WW) 查询 @) 项 目 E) 调式 0) 工具 G) 

窗口 如 ) 社区 人 C) 帮助 00 

也 利 建 查询 思 | 由 | 访 遍 丁目 | 启 加 本 | 总 天 

Wy QW | cheters | 执行 O) 更 v 33 站 
SQLQueryl1. s... ator (53))* 

日 USE chapcer5 

DD INSERT INTO gt 
SELECT name,p' 
FROM gameaceo 


跑 得 快 ”112233 
新 复制 的 数据 
二 


MLSERVER2O08 (.. | WIDN-9527\Adninistrato. 
就 续 行 6 


图 5.11 复制 gameaccount 表 中 的 数据 图 $.12 gameaccountl 表 中 的 数据 


[| YI-952TWAdninistrate .| chapter5 |00;00:00 |6 行 
就 结 行 4 列 ! Ch1i 用 


shapter5 | 00:00:00 |0 行 
列 1 chi 用 


5.1.7 一 次 多 添加 几 条 数据 


在 前 面 的 几 个 小 节 中 ， 也 同时 向 表 中 加 入 过 多 条 记录 。 但 是 ， 实 际 上 只 是 批量 地 执行 
INSERT INTO 语句 。 同 样 的 操作 ， 却 要 重复 地 写 语句 既 浪费 时 间 又 影响 了 数据 库 执行 的 性 
能 。 那 么 ， 有 没有 办 法 只 用 一 条 语句 ， 就 能 进行 批量 地 增加 数据 呢 ? 肯定 有 啊 ， 记 住 下 面 
的 语法 规则 就 可 以 一 次 添加 多 条 数据 了 。 

INSERT INTO table name (column namel, column name2,...) 

VALUES (valuel,value2,value3,...), 

(valuel,value2,value3,...), 


其 中 , 在 VALUES 后 面 列 出 要 添加 的 数据 ， 每 条 数据 之 间 用 “,” 隔 开 就 可 以 了 。 下 


面 就 用 示例 6 来 演示 如 何 一 次 添加 多 条 数据 。 
【示例 6】 向 游戏 账号 信息 表 (gameaccount) 中 添加 表 5-6 中 的 数据 。 
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表 5-6 ”游戏 账号 信息 表 添 加 的 数据 
账号 备注 
5 - 枝 独 秀 | 112233 | 1 初级 联盟 yzdx@126.com 加 油 
6 天 外 飞 仙 | 213456 gE] 高 级 联盟 500 twfx@126.com 快 升级 了 
7 | 条 条 能 | 876986 | 1 | 初级 联盟 | 150 | bbx@126.com | 快 升级 了 


根据 表 5-6 中 的 数据 ， 向 gameaccount 表 中 添加 数据 的 语句 如 下 所 示 。 


USE chapter5 
INSERT INTO gameaccount 


VALUES (5, ' 一 枝 独 秀 ', '112233' ,1, ' 初 级 联盟 ' ,100, 'yzdx@126.com', ' 加 油 ')， 
(6, ' 天 外 飞 仙 ', '213456', 3, ' 高 级 联盟 ', 500, 'twfx@126.com', ' 快 升级 了 ')， 
(7, ' 笨 年 能 ','876986' ,1, ' 初 级 联盟 ' ,100, 'bbx@126.com',' 快 升级 了 '); 


执行 上 面 的 语句 ， 就 可 以 一 次 向 gameaccount 表 添 加 3 条 数据 。 执 行 效果 如 图 5.13 所 示 。 


oft SQL Server Wanagenent Studie 
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二 Laeryl .tor (G3))*| TI-WAW. emerccountl | MIM-G527W nevccounti# | -x 

BUVSE chapter5s 

加 INSERT INTO game 

VALUES (5, ' 一 枝 独 

6,' 天 外 飞 仙 ' 

?， 采 繁 能 ， 


233' ,1,' 初 级 联盟 ' ,100,' yzdx8126.com', ' 加 油 ' 
456' ,3,' 高 级 联盟 ' ,500,'twfx8126.com',' 快 升级 了 ')， 
'876986' ,1,' 初 级 联盟 ', 100, 'bbx8126.com', ' 快 升级 了 ' 


| 

(3 行 受 影响 ) 了 
J 虽 
园 查询 已 成 功 执行 . | TY-952TWISSQLSERYEE2006 〔 . ，| WIDW-9527\Adninistrato.. ，| chapter5 | 00:00:00 | 0 行 
就 绪 hy 列 7 Ch7 Ins /A 


图 5.13 ”向 表 gameaccount 中 添加 多 条 数据 


通过 批量 添加 数据 的 语句 添加 数据 ， 与 每 条 数据 都 使 用 INSERT INTO 语句 添加 效果 
一 样 的 。 查 询 gameaccount 表 数 据 ， 效 果 如 图 5.14 所 示 。 


加 
Er 编辑 E) 视图) 查询 @) 项 目 E) 调 区 0) 工具 CD) 家 OW 社区 加 
帮助 00 

了 新建 查询 | 出 | 也 也 可 记 | 所 四 本 | 怠 瑟 

| tar5 -| ?执行 0 》 mv 弹 忆 回 | ” 盐 | 妈 加 好 
‘SQLQueryl. s... ator (53))*| WIDW-S52TWN emeaccountl | = 


日 USE chapters; 
select * from gameaccount; 批量 添加 的 3 条 数据 


112233 
123456 


987654 
123456 
112233 
213456 
876986 


回 查 -。 | WIDw-9527WRSSQLSERVER2008 ( .| WIDW-9527\Adninistrato... | chapter5 |00:00:00 |7 行 
就 绪 第 1 行 第 1 列 Ins /4 


图 5.14 批量 添加 数据 后 gameaccount 表 的 数据 
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5.2 修改 表 中 的 数据 一 UPDATE 


如 果 房 子 装修 好 后 ， 你 觉得 材料 不 好 或 者 是 颜色 不 好 就 需要 更 换 了 。 数 据 表 中 的 数据 
也 是 一 样 的 ， 如 果 发 现 数据 出 现 了 问题 也 是 允许 修改 的 。 数 据 表 中 的 数据 也 是 一 样 的 ， 当 
发 现存 放 的 数据 不 符合 要 求 时 ， 就 需要 修改 数据 了 。 


5.2.1 UPDATE 语句 的 基本 语法 形式 


修改 数据 表 中 的 数据 使 用 的 是 UPDATE 语句 ,修改 数据 表 的 语句 也 有 很 多 种 形式 。 这 
里 ， 只 给 出 一 个 通用 的 形式 以 便 读者 学 习 。 其 余 的 形式 将 在 下 面 的 小 节 中 详细 讲解 。 修 改 
数据 表 的 一 般 语 法 如 下 : 
UPDATE table name 
SET column namel=valuel, column name2=value2 … 
WHERE conditions 
其 中 : 
口 table name: 表 名 。 要 修改 的 数据 表 表 名 ， 通 常 要 先 打开 该 数据 表 所 在 的 数据 库 。 
请 读者 注意 ， 一 次 只 能 修改 一 张 表 。 
口 column_name: 列 名 。 需 要 修改 列 的 名 字 ，value 是 指 给 列 设置 的 新 值 。 
口 conditions: 条 件 。 按 条 件 有 选择 地 更 新 数据 表 中 的 数据 。 如 果 省 略 了 该 条 件 ， 也 
就 是 省 略 了 WHERE 语句 ， 就 代表 修改 数据 表 中 的 全 部 记录 。 


5.2.2 ”将 表 中 的 数据 全 部 修改 


在 上 面 的 修改 数据 表 的 语法 中 ， 去 掉 WHERE 语句 就 可 以 了 。 很 简单 吧 。 但 是 ， 修 改 
表 中 的 全 部 数据 可 不 是 一 个 常用 的 操作 哦 ， 就 好 像 把 我 们 装修 好 的 房子 重新 装修 了 。 尽 管 
不 常用 ， 还 是 要 学 习 吧 。 下 面 就 用 示例 7 来 演示 一 下 如 何 修改 表 中 的 全 部 数据 吧 。 先 声明 
一 下 ， 所 谓 修改 表 中 的 全 部 数据 ， 只 是 没有 修改 条 件 而 不 是 指 一 定 要 把 表 中 的 所 有 列 的 数 
据 都 修改 了 ! 

【示例 7】 修改 游戏 账号 信息 表 (gameaccount) 中 的 备注 信息 列 (remark)， 将 其 全 部 
修改 成 “无 ”。 

根据 题目 要 求 ， 只 是 修改 游戏 账号 信息 表 的 一 列 ， 也 就 是 在 UPDATE 中 的 SET 语句 
后 只 有 一 个 列 。 具 体 的 语句 如 下 : 

USE chapter5 


UPDATE gameaccount 
SET remark=' 无 '; 


执行 上 面 的 语句 ， 就 可 以 将 游戏 账号 信息 表 中 remark 列 的 值 修改 成 了 “无 >。 具体 的 
执行 效果 如 图 5.15 所 示 。 

从 图 5.15 中 ， 可 以 看 出 通过 上 面 的 修改 语句 ， 修 改 了 表 中 的 7 条 记录 。 修 改 后 如 何 查 
看 修改 的 结果 呢 ? 没 错 ， 还 是 老 办 法 ， 用 SELECT 语句 就 可 以 了 。 游 戏 账号 信息 表 修改 后 
的 效果 如 图 5.16 所 示 。 
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新 机 | 六 | 弟 谤 久 店 芒 回 马 | 加 局 
ae sd| 有 | chepter5 -| 1 执行 有 v 嗓 鲁 国 站 强 
文件 四 编 邵 下， 视图 查询 @) 二 
SQLQuery2. s... ator (56))*| a 
项 目 @) 调试 @) 工具 中 窗口 四 USE chapters; 


社区 外 帮助 中 | SELECT * FRON gameac， 
也 新建 查询 @ | 出 | 也 也 史记 | 蕊 中 
My 凤 | -hapters 4 


game121@1 
yedk@126c 


| YID-9527\MISSQLSERVER2008 〔 .. | YTDY-9527\WAdninistrate .. | chapter5 |00:00:00 | 7 行 
后 ttF9) 已 按 下 。 行 4 列 1 Ch1 x 


图 5.15 ”修改 游戏 账号 信息 表 中 的 备注 信息 图 5.16 ”修改 游戏 账号 信息 表 remark 列 的 效果 


通过 图 5.16 就 可 以 一 日 了 然 了 吧 , 上 面 的 修改 语句 确实 是 将 游戏 账号 信息 表 中 remark 
列 全 部 修改 成 了 “无 >。 请 读者 自己 尝试 一 下 ， 将 游戏 账号 信息 表 的 所 有 的 积分 (points) 
和 等 级 〈level) 列 分 别 修改 成 0 和 1。 


5.2.3 ”只 修改 想 要 修改 的 数据 


所 谓 只 修改 想 要 修改 的 数据 ， 就 是 指 想 按照 什么 条 件 修 改 哪 条 记录 就 修改 哪 条 记录 ， 
比如 : 将 积分 在 100 以 下 的 都 修改 成 100， 将 等 级 是 1 级 的 都 修改 成 2 级 等 。 如 果 想 按 条 
件 修改 数据 表 中 的 数据 ， 就 需要 在 UPDATE 语句 中 使 用 WHERE 子 名 了。 下 面 就 用 示例 8 
来 演示 WHERE 子 句 是 如 何 使 用 的 。 

【示例 8】 将 游戏 账号 信息 表 (gameaccount) 中 的 账号 等 级 是 1 的 全 部 修改 成 2。 

根据 题目 的 要 求 ， 修 改 游戏 账号 信息 表 的 条 件 只 有 一 个 ， 并 且 要 修改 的 列 只 有 账户 等 
级 列 。 具 体 的 修改 语句 如 下 : 

USE chapter5; 

UPDATE gameaccount 

SET level=2 

WHERE level=1; 

通过 上 面 的 语句 ， 就 可 以 将 游戏 账号 信息 表 中 账号 等 级 是 1 的 全 部 修改 成 2 了。 执行 
效果 如 图 5.17 所 示 。 

从 图 5.17 中 ,可 以 看 出 执行 上 面 的 修改 语句 后 ,修改 了 账号 信息 表 中 的 4 条 记录 。 查 
询 游戏 账号 信息 表 ， 效 果 如 图 5.18 所 示 。 

读者 看 了 图 5.18 的 查询 结果 后 , 可 能 会 有 这 样 一 个 疑问 吧 ? 在 游戏 账号 信息 表 中 现在 
一 共有 5 条 记录 的 等 级 都 是 2， 如 何 知道 修改 语句 执行 成 功 了 呢 ? 呵呵 ， 其 实 这 个 并 不 难 
啊 ， 可 以 换个 角度 看 ， 如 果 游 戏 账号 信息 表 中 没有 等 级 是 1 的 记录 了 ， 那 么 不 就 说 明 修改 
成 功 了 嘛 。 


a 
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图 5.17 修改 游戏 账号 信息 表 中 账号 等 级 是 1 的 记录 图 5.18 修改 账号 等 级 后 的 效果 


5.2.4 修改 前 N 条 数据 


如 果 在 修改 数据 时 ， 想 完成 把 账号 等 级 是 2 的 记录 只 修改 1 条 ， 应 该 怎么 办 呢 ? 仔细 
想 一 下 ,用 前 面 给 出 的 UPDATE 语句 形式 是 无 法 完成 的 吧 。 不 要 紧 ， 现在 就 告诉 你 修改 的 
方法 。 用 TOP 关键 字 ! 具体 的 语法 形式 如 下 : 

UPDATE TOP (n) table name 

SET column namel=valuel, column name2=value2... 

WHERE condition; 

这 里 ，TOP (n) 中 的 n 是 指 前 儿 条 记录 ， 是 一 个 整数 。 其 余 的 语句 都 在 前 面 的 UPDATE 
语句 中 解释 过 了 ， 这 里 就 不 再 袭 述 了 。 下 面 就 请 读者 一 起 通过 示例 9 来 体验 一 下 如 何 使 用 
TOP 关键 字 吧 。 

【示例 9】 修改 游戏 账号 信息 表 (gameaccount) 中 等 级 是 2 的 前 2 条 记录 ， 将 其 备注 
信息 修改 成 “这 是 刚 修改 过 的 ”。 

根据 题目 要 求 ， 该 语句 中 应 该 用 WHERE 子 句 来 限制 条 件 ， 并 用 TOP(2) 来 确定 修改 2 
条 记录 。 修 改 语句 如 下 : 

USE chapter5; 


UPDATE TOP(2) gameaccount 
SET remark=' 这 是 刚 修 改过 的 ' 


WHERE level=2; 


执行 上 面 的 语句 ， 就 可 以 将 level 为 2 的 前 2 条 记录 修改 了 。 执 行 效果 如 图 5.19 所 示 。 

这 回 为 了 能 够 让 修改 有 一 个 直观 的 效果 , 特地 将 remark 字段 修改 成 了 “这 是 刚 修改 过 
的 ”查询 修改 后 的 游戏 账号 信息 表 ， 效 果 如 图 5.20 所 示 。 下 面 就 请 读者 在 图 5.20 所 示 的 
效果 中 找到 修改 的 2 条 记录 。 


5.2.5 ”根据 其 他 表 的 数据 更 新 表 
上 面 儿 个 修改 数据 表 的 例子 都 是 通过 自己 指定 数据 来 修改 的 ， 如 果 你 要 修改 的 数据 在 
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编辑 下 ) 视图 WD 查询 @) 项 目 f) 
调试 @j) 工具 0) 窗口 OD 社区 加 帮助 0D 


了 98 如 | 访 | 戌 访 凶 | 访 | 区 回电 久 


gamear cE 
SET remark=' 这 是 刚 修改 过 的 ' 


HWHERE level=2: 


MTDW-952T\WAdaninistrate . .| chapter5 | 00:00:00 |0 行 


就 绪 行 6 列 1 Ch1 
图 5.19 修改 游戏 账号 信息 表 中 等 级 为 2 的 前 2 条 记录 

Amicrosoft SQL Server Nanagement Studio -Ilolxl 
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帮助 00) 

ETETETT 天 

Wy 8 ER -| 执行) 》 sv 3 昌国 | 于 克 | 锥 国共 各 
“SQLQuery2. s. - -ater (56))*#| vx 
章 US chapters; 

| select * from gameaccount; 习 
«| 1 » 


| name | password gamegroup | pont: 
112233 pao@126 com ET 
123456 NULL 
39987654 paopaoG126com 无 
123456 game121@126.com 无 
112233 yedx@126 com 
213456 Iwix@126 com 
876986 bbw@126 com 


.| WIDOW-g527\Adninistrato. .. | chapter5 | 00:00:00 |7 行 
行 4 列 ! Chl Ins /4 


图 5.20 ”修改 游戏 账号 信息 表 后 的 效果 


数据 库 中 的 另 一 张 表 中 已 经 存在 了 ， 可 不 可 以 直接 复制 过 来 使 用 呢 ? 当然 是 可 以 的 ， 但 是 
这 可 不 像 在 Excel 表格 中 复制 数据 那么 简单 。 如 果 读 者 要 想 完成 从 另 一 个 表 复 制 数据 的 操 
作 ， 就 要 记 住 如 下 的 语句 了 。 

UPDATE table name 

SETcolumn namel=tablel name.column namel, 

column name2=tablel name.column name2... 

FROM tablel name 

WHERE conditions 

其 中 : 
table name: 表 名 。 要 更 新 的 数据 表 名 称 。 
column_name1: 列 名 。 要 修改 数据 表 中 的 列 名 。 
tablel_name: 表 名 。 要 复制 数据 的 数据 表 名 称 。 
tablel_name.column_namel: 列 名 。 要 复制 数据 的 列 ， 记 住 ， 一 定 要 在 列 名 前 面 加 
上 表 名 。 
口 conditions: 条 件 。 它 是 指 按 什 么 条 件 来 复制 表 中 的 数据 。 
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记 住 了 上 面 的 语法 规则 ， 就 可 以 在 修改 数据 表 时 使 用 其 他 表 中 的 数据 了 。 下 面 就 用 示 
例 10 来 实际 演练 一 下 吧 。 在 演练 之 前 要 先 创建 一 张 数据 表 并 存 入 一 些 数据 ,以 便 复 制 数据 
使 用 哦 。 创 建 一 张 游戏 账号 等 级 信息 表 ， 表 的 结构 如 表 5-7 所 示 。 


表 5-7 游戏 账号 等 级 信息 表 (gamelevel) 


字 段 名 数 据 类 型 描 述 
1 id int 等 级 编号 
eA level 等 级 
3 points 积分 
上 面 的 游戏 账号 等 级 信息 表 主 要 描述 了 每 种 等 级 对 应 的 积分 情况 。 根 据 上 面 的 表 结 构 
创建 数据 表 的 语句 如 下 : 
USE chapter5; 
CREATE TABLE gamelevel 
( 
id INT PRIMARY KEY identity(1,1), 
level INT, 
points INT, 
Ne 
执行 了 上 面 的 创建 数据 表 的 语句 后 ， 再 为 其 添加 如 表 5-8 所 示 的 数据 。 我 们 要 复制 的 
数据 来 源 就 创建 好 了 。 


局 
uj 


表 5-8 ”游戏 账号 等 级 信息 表 添加 的 数据 


编号 (id) 等 级 (level) 积分 (points) 
1 1 100 
2 2 500 
3 3 1000 


如 果 读 者 是 按 顺 序 阅 读本 章 内 容 的 ， 那 么 ， 对 于 批量 向 数据 表 中 添加 数据 的 操作 是 不 
会 陌生 的 。 下 面 就 来 操练 一 下 吧 。 具 体 的 语句 如 下 : 

INSERT INTO gamelevel 

VALUES (1,100), (2,500), (3,1000)»; 

执行 上 面 的 语句 后 ， 游 戏 账号 等 级 信息 表 中 就 有 3 条 数据 了 。 

【示例 10】 将 游戏 账号 信息 表 中 的 积分 信息 按照 游戏 账号 等 级 信息 表 中 每 个 等 级 对 应 
的 积分 进行 更 新 。 

根据 题目 的 要 求 ， 要 更 新 游戏 账号 信息 表 中 的 积分 信息 ， 更 新 的 语句 如 下 : 

USE chapter5; 

UPDATE gameaccount 

SET points=gamelevel .points 

FROM gamelevel 

WHERE gameaccount.level=gamelevel.level; 

执行 上 面 的 语句 ， 就 可 以 将 游戏 账号 信息 表 中 的 积分 根据 游戏 账号 等 级 信息 表 进 行 更 
新 了 。 执 行 效果 如 图 5.21 所 示 。 
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文件 作 编辑 下 ) 视图 查询 @) 项 目 @) 调式 @) 工具 C) 窗口 如 


社区 EC) 帮助 中 


;也 新建 查询 轨 | 记 | 思 也 怠 巴 | 记 回 权 驴 下 
; Be 28 | aapters -| 1 执行 @) 》 和 x 中 量 国 | 下 名 


SQLQueryl. s... ator (56))#| 


USE chapter5: 

UPDATE gameaccount 

SET points=gamelevel.points 
FRON gamelevel 


已 消息 | 


WHERE gameaccount.level=gamelevel. level; 


(5 行 受 影响 ) 
加 喇 
lio MssQLSERYER2008 〔 . ，| WIDW-9527\Adninistrato. .| chapters |00:00:00 |0 行 
就 绪 行 6 Ch1 


图 5.21 更 新 游戏 账号 信息 表 中 的 积分 


从 图 5.21 所 示 的 执行 效果 中 ， 可 以 看 出 游戏 账号 信息 表 中 的 6 行 数据 已 经 被 更 新 了 。 


更 新 后 的 游戏 账号 信息 表 中 的 数据 如 图 5.22 所 示 。 


Raicrosoft SQL Server Wanagenent Studio 


jx 


文件 时 ) 编辑 人 E) 视图 查询 @) 项 目 @) 调试 @) 工具 C) 窗口 i 社区 人 C) 


帮助 0 


ET 


MY Wa | chapter5 "| 执行) 有 时 剖 国 |” 列 | 的 名 


SQLQueryl. s... ator (56))#| 


日 USE chapters; 
| SELECT * FRON gameaccount; 


| WIDW-9527 MSSQLSERVER2008 ( .，| WIDW-9527\Adninistrato. .. | chapter5 | 00:00:00 |7 行 


就 绪 行 5 列 1 


图 5.22 游戏 账号 信息 表 更 新 后 的 效果 


从 图 5.22 中 可 以 看 出 ， 在 游戏 账号 信息 表 中 ， 积 分 (points) 列 中 有 一 个 值 是 10， 该 
值 为 什么 没有 被 更 新 呢 ? 找到 答案 了 吗 ， 没 错 ， 就 是 因为 该 行 中 等 级 〈level) 是 0， 在 游 


戏 账号 等 级 信息 表 中 没有 这 个 等 级 哦 ! 


5.3 使 用 DELETE 语句 删除 表 中 的 数据 


如 果 房 子 装修 好 后 ， 觉 得 有 些 东 西 实在 是 不 太 合适 或 者 是 不 喜欢 了 ， 也 是 可 以 直接 拆 


掉 或 者 扔 掉 的 吧 。 数 据 表 中 的 数据 也 是 一 样 的 ， 如 果菜 些 数 据 已 经 没有 作用 了 ， 就 可 以 将 其 


去 除了 。 众 所 周知 ， 扔 东西 是 最 简单 的 了 。 那 么 ， 删 除数 据 表 中 的 数据 也 是 最 简单 的 了 。 但 


是 ， 要 注意 哦 ， 删 除 的 数据 就 不 那么 容易 恢复 了 ， 就 像 扔 了 的 东西 不 容易 找 
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回来 是 一 样 的 。 
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5.3.1 DELETE 语句 的 基本 语法 形式 


在 删除 数据 表 中 的 数据 前 ， 如 果 不 能 确定 这 个 数据 以 后 是 否 还 会 有 用 ， 一 定 要 对 其 数 
据 先 进行 备份 哦 ! 删除 数据 表 中 的 数据 使 用 的 语法 很 简单 ， 一 般 的 语法 形式 如 下 : 

DELETE FROM table name 

WHERE conditions; 

其 中 : 

口 table name: 表 名 。 要 删除 数据 的 数据 表 名 称 。 

口 conditions: 条 件 。 按 照 指 定 条 件 删除 数据 表 中 的 数据 ， 如 果 没 有 指定 删除 条 件 就 

是 要 删除 表 中 的 全 部 数据 了 。 
另外 ， 在 上 面 的 语句 中 也 可 以 将 DELETE 后 面 的 FROM 关键 字 省 略 。 


5.3.2 ”清空 表 中 的 数据 


所 谓 清空 表 中 的 数据 ， 就 是 删除 表 中 的 全 部 数据 。 换 句 话说， 就 是 在 使 用 删除 语句 时 
不 加 WHERE 子 句 。 下 面 就 用 示例 11 来 演示 一 下 如 何 清空 表 中 的 数据 。 

【示例 11】 将 游戏 账号 等 级 信息 表 中 的 数据 全 部 删除 。 

根据 题目 的 要 求 ， 删 除 游戏 账号 等 级 信息 表 〈gamelevel) 中 的 数据 语句 如 下 : 

USE chapter5; 

DELETE FROM gamelevel; 

执行 上 面 的 语句 ， 游 戏 账 号 等 级 信息 表 中 的 数据 就 消失 了 。 执 行 效果 如 图 5.23 所 示 。 

从 图 5.23 的 执行 结果 中 ， 可 以 看 到 已 经 有 3 行 数据 被 删除 了 。 删 除 后 就 剩 下 一 张 空 表 
了 ， 效 果 如 图 5.24 所 示 。 


Ricrosoft SQL SAFE 

文件 E) 编辑 到 ) 视图 WD 查询 @) 
项 目 中 ) 调试 @) 工具 CD) 窗口 中 
社区 CC) 帮助 0) 

辽 闽 建 查 询 中 | 六 | 孔 也 本 记号 


编辑 下 ) 视图) 查询 @) 
项 目 @) 调式 @) 工具 CD) 窗口 岂 ) 


By Wa | chepters 
SQLQueryl. s... ator (56))*| 
USE chapters 


| DELETE FROM gemelevel 


4 
已 消息 | 


行 受 影响 ) 2 


WAdninistrate .. | shapter5 | 00:00:00 |0 行 
就 绪 4 ch | 


.| chapters |00:00:00 |0 行 
行列 Ch 


图 5.23 ”加 除 游戏 账号 等 级 信息 表 中 的 数据 图 5.24 清空 后 的 游戏 账号 等 级 信息 表 


从 图 5.24 中 ,可 以 看 出 虽然 将 游戏 账号 等 级 信息 表 中 的 数据 删 掉 了 ， 但 是 表 的 结构 还 
是 保留 的 。 


5.3.3 ”根据 条 件 去 掉 没 用 的 数据 


在 实际 应 用 中 ， 删 除 表 中 全 部 数据 的 操作 是 不 太 常 用 的 ， 就 像 刚 装修 的 房子 全 部 拆除 
的 可 能 性 是 比较 小 的 。 但 是 ， 不 要 忘 了 删除 表 中 数据 的 语法 中 可 以 有 WHERE 语句 啊 ， 这 
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就 是 按 条 件 删除 数据 。 下 面 就 在 示例 12 中 学 习 一 下 如 何 按 条 件 删除 表 中 的 数据 吧 。 
【示例 12】 将 游戏 账号 信息 表 中 等 级 是 0 的 记录 删 掉 。 
根据 题目 要 求 ， 只 删除 等 级 是 0 的 记录 ， 删 除 语句 如 下 : 
USE chapter5; 


DELETE FROM gameaccount 
WHERE level=0; 


执行 上 面 的 语句 ， 就 可 以 将 游戏 账号 信息 表 中 等 级 是 0 的 记录 删 掉 了 。 执 行 效果 如 
图 5.25 所 示 。 

从 图 5.25 中 可 以 看 出 ,影响 了 表 中 的 1 条 记录 ,也 就 是 原来 表 中 只 有 1 条 记录 等 级 是 
0。 删 除 后 查看 游戏 账号 信息 表 ， 效 果 如 图 5.26 所 示 。 
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9 S 二 | 口交 
文件 下) 编 缉 世 ) 视图) 查询 @) 项 目 E) 调试 @) 工具 C) 窗口 岂 社区 CC) 
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也 查询 四 | 让] 也 也 可 | 户 昌 USE chapters; 

od | chepters 一 | SELECT * FROM gameaccounc; | ， 
SQLQuery2. 5... ator (54))#*| vw 

日 USE chapcer5 

由 DELETE FROM gearweaccount | password [ level | gamegoup | pomits | emai 

fvHERE level-0; 112233 初级 联盟 。 500 。 pao@126 com 这 是 刚 修改 过 的 


2 

123456 2 中 级 联盟 500 NULL 这 是 刚 修改 过 的 
987654 2 。 初级 联盟 500 。 paopao@12.， 无 

112233 ”2 初级 联盟 500 yzdx@126c.， 无 

213456 ”3 高 级 联盟 1000 twix@126com 无 

876986 2 初级 联盟 500 bbx@1l26com 无 


| 


人 1 行 受 影响 ) 
Ld 


@ | II-ss27 ssQLSERYER2008 (.. | WIDN-9527\Adninistrato. .. | chapter5 |00:00:00 |6 行 
行 5 列 1 Ch 1 ee 


图 5.25 删除 等 级 是 0 的 记录 图 526 ”删除 等 级 是 0 的 记录 后 的 效果 
读者 在 图 5.26 中 没有 找到 等 级 是 0 的 记录 吧 ! 没 错 ， 已 经 被 删 掉 了 。 
5.3.4 删除 前 N 条 数据 


先 让 读者 思考 一 个 问题 吧 ， 如 果 想 删除 2 条 等 级 是 2 的 游戏 账户 信息 ， 怎 么 办 呢 ? 回 
忆 之 前 学 习 过 的 DELETE 语句 的 语法 形式 , 没有 能 够 满足 要 求 的 语句 。 除 非 在 游戏 账号 信 
息 表 中 只 有 2 条 是 等 级 2 的 记录 , 才 可 以 通过 在 DELETE 语句 中 的 WHERE 语句 里 设置 条 
件 来 删除 。 这 种 巧合 的 事情 发 生 的 概率 太 小 了 吧 。 如 果 你 没有 这 么 幸运 ， 那 就 继续 向 下 学 
习 吧 。 请 看 如 下 的 语法 ， 它 会 帮助 你 哦 。 


DELETE TOP (N) table name 
WHERE conditions; 


其 中 : TOP(N) 就 是 用 来 指定 删除 前 N 条 记录 的 。TOP(2) 就 是 指 前 2 条 记录 。 
下 面 就 通过 示例 13 来 解答 刚才 给 读者 出 的 思考 题 。 

【示例 13】 从 游戏 账户 信息 表 中 删除 2 条 等 级 是 2 的 游戏 账户 信息 。 
根据 题目 要 求 ， 删 除 2 条 记录 使 用 TOP(2) 就 可 以 了 ， 删 除 语句 如 下 : 


USE chapter5; 
DELETE TOP(2) gameaccount 
WHERE level=2; 


le 
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执行 上 面 的 语句 ， 就 可 以 删除 2 条 等 级 是 2 的 记录 了 。 执 行 效 果 如 图 5.27 所 示 。 
从 图 5.27 中 ， 可 以 看 出 结果 是 “2 行 受 影响 ”， 也 就 是 仅 删除 了 2 条 等 级 是 2 的 记录 。 
删除 后 查询 游戏 账号 信息 表 ， 效 果 如 图 5.28 所 示 。 
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My a | chapters ”| ?执行 C) 三 
SQlQuery2. sater (54))* 
BUSE chapcer57 
由 DELETE TOP (2) gameaccount 
们 2 level=2; 


日 USE chapters; 
SELECT * FRON gameaccount; 


快 快 跑 987654 paopao@126.com 无 


1 
巧 消 息 | | 回避 5 一 梳 。 112233 yedx@126com ”无 

时 | | 6 天 外 213456 twix@126.com 无 
be 行 受 影响 ) 本 | 7 条 每 能 ”876988 bbx@126com 无 
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行 4 列 ! ch 1 区 
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图 5.27 删除 2 条 等 级 是 2 的 记录 图 5.28 删除 2 条 等 级 是 2 的 记录 后 的 效果 


只 看 图 5.28 还 看 不 出 删除 了 哪 2 条 记录 ， 请 读者 对 比 图 5.26 的 效果 ， 找 到 删除 的 是 
哪 2 条 记录 。 


5.3.5 使 用 TRUNCATE TABLE 语句 也 能 清空 表 中 的 数据 


前 面 已 经 说 过 在 清空 表 中 的 数据 时 ， 可 以 使 用 DELETE 语句 来 完成 。 实 际 上 ， 除 了 使 
用 DELETE 语句 外 , 还 可 以 使 用 TRUNCATE TABLE 语句 来 清空 表 中 的 数据 . TRUNACTE 
TABLE 语句 的 语法 很 简单 , 只 需要 在 其 后 面 加 上 表 名 就 可 以 了 。 具 体 的 语法 形式 如 下 所 示 。 


TRUNCATE TABLE table name; 


其 中 ，table_name 就 是 表 名 ， 同 样 在 删除 之 前 还 要 打开 表 所 在 的 数据 库 。 

【示例 14】 使 用 TRUNCATE TABLE 语句 删除 游戏 账号 信息 表 (gameaccount) 中 的 
数据 。 

根据 题目 要 求 ， 删 除 游戏 账号 信息 表 中 的 数据 ， 语 句 如 下 : 


TRUNCATE TABLE gameaccount; 


执行 上 面 的 语句 , 就 可 以 将 游戏 账号 信息 表 | A 于 人 rg ‘=I9lx| 
中 的 数据 全 部 删除 了 。 执行 效果 如 图 5.29 所 示 。 IRD anm EO WhO 

从 图 5.29 的 执行 效果 可 以 看 出 ， 在 执行 了 。 el EE 
TRUNCATE TABLE 语句 后 并 没有 像 执行 oe paperes 
DELETE 语句 时 会 出 现 < 影 响 儿 行 ” 的 消息 提示 ， 下 三 汪汪 
而 是 出 现 “ 命 令 已 成 功 完成 ”的 消息 提示 了 。 很 
显然 ，TRUNCATE TABLE 语句 就 和 DELETE 
语句 不 是 一 种 类 型 的 语句 ， 读 者 可 以 回忆 一 下 ， 
在 什么 时 候 你 见 过 “命令 已 成 功 完成 ”的 这 种 消 ”图 5.29 删除 游戏 账号 信息 表 中 的 全 部 数据 


2008 ( .|WIM-9S27\Adninistrato. .| chapter5 |00:00:00 |0 行 
行 4 列 1 


ED 
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息 提 示 呢 ? 请 读者 翻 到 本 书 的 第 2 章 或 者 第 3 章 ， 一 定 会 知道 类 似 的 消息 提示 。 没 错 ， 实 际 

上 ，TRUNCATE TABLE 也 属于 数据 定义 语言 ， 与 CREATE 、DROP 和 ALTER 是 一 类 的 。 
通过 上 面 的 示例 相信 读者 已 经 了 解 了 TRUNCATE TABLE 的 基本 用 法 ， 那 么 ， 

TRUNCATE TABLE 与 DELETE 清空 表 中 的 数据 有 什么 区 别 呢 ? 主要 区 别 有 两 个 ， 具 体 说 

明 如 下 。 

1. 使 用 TRUNCATE TABLE 删除 数据 速度 较 快 


由 于 使 用 DELETE 语句 删除 数据 时 ,需要 把 删除 的 信息 写 入 到 事务 日 志文 件 中 ， 这样 
能 够 编译 恢复 删除 的 数据 。 而 使 用 TRUNCATE TABLE 语句 删除 数据 ， 是 不 会 将 删除 的 信 
息 写 入 事务 日 志文 件 的 。 因 此 ， 使 用 TRUNCATE TABLE 删除 数据 的 速度 较 快 。 当 表 中 的 
数据 不 再 需要 恢复 时 ， 可 以 使 用 TRUNCATE TABLE 语句 来 完成 删除 操作 。 


2. 使 用 TRUNCATE TABLE 删除 数据 后 自 增长 字段 重新 编号 


在 使 用 TRUNCATE TABLE 语句 清空 表 中 的 所 有 记录 后 ， 表 中 的 自 增长 字段 会 重新 编 
号 ， 而 使 用 DELELE 删除 表 中 的 全 部 记录 后 ， 表 中 的 自 增 长 字段 会 继续 累加 。 为 了 验证 
TRUNCATE TABLE 与 DELETE 删除 数据 后 对 自 增长 字段 的 影响 ， 新 创建 一 张 数 据 表 
TEST， 表 结构 如 表 5-9 所 示 。 

创建 好 TEST 表 后 ， 向 表 中 添加 如 表 5-10 所 示 的 数据 。 

创建 TEST 表 并 添加 数据 的 语句 如 下 : 

USE chapter5; 


CREATE TABLE TEST 
( 
id int identity(1,1), 
name varchar (20) 
); 
INSERT INTO TEST VALUES(' 桌 子 '),(' 椅 子 '),(' 书 ');， 


执行 上 面 的 语句 ，TEST 表 中 的 数据 如 图 5.30 所 示 。 


=I9)x 
表 5-9 TEST 表 结 构 文件 中。 名 各 外 视图) 查询 四 ) 
项 目 @) 调试 @) 工具 上 ) 窗口 @) 
和 区 加 帮助 中 


卫 新 建 查 询 | 记 | 孔 也 中 到 


表 5-10 TEST 表 添 加 的 数据 


编号 (id) 名 称 (name) 
1 桌子 
2 椅子 
3 书 图 5.30 TEST 表 中 的 数据 


下 面 分 别 使 用 TRUNCATE TABLE 语句 和 DELETE 语句 删除 TEST 中 的 数据 。 
(1) 使 用 TRUNCATE TABLE 语句 删除 TEST 表 中 的 数据 
在 删除 之 前 ， 记 住 在 图 5.30 中 ,id 是 自 增长 列 并 且 序 号 已 经 变 成 3 了 。 删 除 语句 如 下 


全 和 
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所 示 。 
USE chapter5; 
TRUNCATE TABLE TEST7 
执行 上 面 的 语句 后 ，TEST 表 中 就 没有 数据 了 。 下 面 就 到 了 最 关键 的 时 候 了 ， 向 TEST 
表 中 再 插入 一 条 数据 ， 语 句 如 下 : 


USE chapter5; 
INSERT INTO TEST VALUES(' 本 '); 


下 面 就 要 揭晓 答案 了 ， 查 询 TEST 表 中 的 数据 ， 看 看 id 究竟 是 几 呢 ? 如 图 5.31 所 示 。 

从 图 5.31 中 ,可 以 看 到 id 值 是 1, 也 就 是 说 使 用 TRUNCATE TABLE 语句 删除 数据 后 ， 
自 增 长 列 是 重新 计数 的 。 

(2) 使 用 DELETE 语句 删除 TEST 表 中 的 数据 

目前 ， 在 TEST 表 中 只 有 一 条 数据 ，id 为 1。 下 面 就 使 用 DELETE 语句 将 其 删除 ， 语 
名 如 下 : 


USE chapter5; 
DELETE TEST; 


执行 上 面 的 语句 ，TEST 表 中 的 数据 也 就 被 删除 了 。 下 面 也 同样 再 向 TEST 表 中 添加 1 
条 记录 ， 看 看 id 的 变化 。 添 加 数据 的 语句 如 下 : 

USE chapter5; 

INSERT INTO TEST VALUES(' 书 包 '); 


好 了 ， 这 回 查 看 TEST 表 ， 看 看 id 是 多 少 了 。 如 图 5.32 所 示 。 


WT olx) Ricrosoft SQL Server Naneeeneaoe =|D|x| 
文件 中 编辑 EE) 视图 WW 查询 @) 项 目下 ) 文件 人 ) 编辑 下 ) 视图 WW 查询 @) 项目 @) 
调 坛 @) 工具 人 窗口 WD 社区 帮助 


调式 W) 工具 四 窗口 WD 社区 C) 帮助 0D 
和 2 查询 如 | 昼 | 说 名 芒 | 局 | 苞 回 四 副 
电 | hote5 | ?执行 Q) 总 
| “SQLQuery2. s... ator (54))* 
USE chapters; 
INSERT INTO TEST VALUES(' 书 包 '); 
SELECT + FROM TE: 


ET 
BY 1 | -hapters ~| ?执行 Q) 》 名 
SQLQuery2. s... ator (54))* 
日 USE chapters; 

INSERT INTO TEST VALUES1' 本 ，) 
SELECT * FRON TEST 


x 


ST; 


日 


，| ID-9527\Adninistrato. . .| chapter5 | 00:00:00 |1 行 MTDY-9527\Wdninistrate .，| chapter5 |00:00:00 |1 行 


职 渚 行 7 列 ! cl /4 就 绪 行 5 列 ! cl 4 


图 5.31 使 用 TRUNCATE TABLE 删除 数据 后 图 5.32 ”使 用 DELETE 删除 数据 后 自 增长 列 
自 增长 列 的 变化 的 变化 


这 回 看 到 id 的 变化 了 吧 , id 的 值 是 2， 也 就 是 说 通过 DELETE 删除 数据 后 ， 自 增长 列 
的 序号 是 在 原来 的 基础 上 继续 增加 的 。 


5.4 使 用 企业 管理 器 操作 数据 表 


经 历 了 前 面 使 用 SQL 语句 对 数据 表 中 数据 的 添加 、 修改 以 及 删除 的 操作 ， 相 信 读 者 
经 厌倦 了 。 这 么 多 的 语句 ， 要 记 住 它们 真 麻 烦 啊 。 现 在 就 教 给 你 一 个 简单 的 方法 吧 ， 使 


斑 避 


“ae 
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企业 管理 器 来 操作 数据 表 。 

企业 管理 器 不 用 多 说 ， 读 者 也 知道 了 ， 它 就 是 一 个 用 鼠标 操作 的 友好 界面 。 使 用 企业 
管理 器 操作 表 中 的 数据 , 可 谓 是 最 能 够 体现 它 的 便利 了 。 下 面 通过 示例 15 来 演示 如 何在 企 
业 管 理 器 中 添加 数据 、 修 改 数据 以 及 删除 数据 。 

【示例 15】 在 企业 管理 器 中 对 游戏 账号 信息 表 做 如 下 操作 。 

(1) 向 数据 表 中 添加 如 表 5-11 所 示 的 数据 。 

(2) 修改 编号 是 1 的 记录 ， 将 其 积分 修改 成 2000。 

(3) 删除 编号 是 1 的 记录 。 


表 5-11 游戏 账号 信息 表 添 加 的 数据 
备注 
| 剪刀 石头 布 | 112233 | 1 | 初级 联盟 | 100 | jzstv@126com | 天 
无 
| 九 必 白狐 | 334455 | 1 | 初级 联盟 | 100 | jwon@I26com | 无 


无 论 对 游戏 账号 信息 表 做 添加 、 修 改 还 是 删除 操作 ， 都 需要 在 游戏 账号 信息 表 的 表 编 
辑 界面 中 完成 。 游 戏 账 号 信息 表 的 编辑 界面 ， 需 要 在 企业 管理 器 里 的 对 象 管理 器 中 ， 展 开 
chapter5 数据 库 ， 并 右 击 该 数据 库 中 的 表 gameaccount， 在 弹出 的 右键 菜单 中 选择 “编辑 前 
200 行 ”选项 ， 即 可 见 到 该 界面 了 ， 如 图 5.33 所 示 。 


mi conent S 
文件 时) 编辑 里 ) 视图 WD) 项 目下 ) 调试 四) 查 海 设计 器 铺 ) 工具 (0) 人 定 口 加 社区 (CC) 帮助 如 
WD 六 | 护 轧 久 DD 区 9 总 s 
习 册 双 周 | ww ? 了 |[5 | 国 奖 


图 5.33 ”游戏 账号 信息 表 的 编辑 界面 


本 题 的 3 个 小 问 ， 都 可 以 在 图 5.33 所 示 的 界面 中 完成 。 下 面 分 别 讲解 如 下 。 
(1) 添 加 表 5-11 所 示 的 数据 , 就 像 在 Excel 表 中 输入 信息 一 样 。 录入 表 5-11 的 信息 后 ， 
效果 如 图 5.34 所 示 。 


Rx icrosoft SQL Server 可 
文件 如 ”编辑 外 视图 0) 项目) 调试 @) 查询 设计 器 @@) 工具 和 ) 窗口 社区 C) 帮助 只 


卫 采 寻 查询 中 | 让 | 钙 迎 镶 | 访 | 瑟 旺 各] 加 5 
BEE SS 


图 5.34 ”向 gameaccount 表 中 添加 数据 


和 
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添加 数据 后 如 何 保存 呢 ? 在 企业 管理 器 中 ， 为 表 添 加 数据 不 用 刻意 地 去 保存 数据 ， 只 
需要 把 光标 移动 到 下 一 行 ， 就 保存 了 上 一 行 的 数据 了 。 


(2) 将 id 是 1 的 记录 ， 积 分 改 成 2000 也 很 简单 。 只 需要 在 图 5.34 所 示 的 界面 中 ， 直 
接 将 id 为 1 的 列 所 对 应 的 points 列 改 成 2000 就 可 以 了 。 效 果 如 图 5.35 所 示 。 


E se . =I9lxl 
2 四 pr ET 8 调试 回 ) ”查询 设计 器 他 ) 工具 CD) 窗口 @ 社区 人 C) 帮助 中 
也 章 建 查询 中 | 记 | 轴 也 加 | 记 | 启 昌 乌 | 受 后 
本 组 双 | 国 | 更 g 关 型 "| ? 码 |[E 避 韶 瑟 
fe i pry ee ms le x 
name |password |level |gamegroup |points |emall | remark 
台 刀 石头 布 112233 1 初 绍 联盟 festb@126.com 无 
寻 欧 223344 2 中 级 联盟 So0 xm@126.com 无 
九 尾 白狐 334455 1 初 凶 联盟 10 jmbh@l26.com 无 
NAL NUL MAL MAL ML ML NAL 
Hj)» Hy © Rm] 


图 5.35 修改 id 为 1 的 points 列 
从 图 5.35 可 以 看 出 ， 当 前 游戏 账号 信息 表 的 状态 是 “单元 格 已 修改 ” 修改 数据 后 
仍然 将 光标 移动 到 其 他 单元 格 中 ， 就 可 以 保存 该 数据 了 。 
(3) 删除 数据 会 稍微 复杂 一 些 。 删 除 编号 是 1 的 记录 ， 需 要 先 单 击 要 删除 的 记录 使 其 
下 于 ; 本 


处 于 选中 状态 ， 然 后 右 击 该 记录 ， 在 弹出 的 右键 菜单 中 选择 “删除 ”选项 ， 弹 出 如 图 5.36 
所 示 的 界面 。 


也 新建 查询 中 | 让 | 负 也 相 | 包 | 放 昌 鸟 四 


S251] 


回国 到 国 | Re 


单 击 “ 是 ”将 永久 删除 这 些 行 。 您 将 无 法 撒 消 所 做 的 更 疏 。 


| 


Il 4|1 13|b MP | | 
就 绪 


图 5.36 删除 提示 界面 
在 图 5.36 所 示 的 界面 中 ， 单 击 “ 是 ”按钮 ， 即 可 将 所 选 的 记录 删除 了 。 


名 说明 : 如 果 要 删除 多 条 记录 ， 不 用 一 条 一 条 地 选择 。 只 需要 按 住 Shift 或 Ctrl 键 的 同时 
选中 要 删除 的 记录 ， 然 后 一 起 删除 就 可 以 了 。 


5.5 本 章 


本 章 主要 讲解 了 如 何 使 用 SQL 语句 和 企业 管理 器 向 数据 表 中 添加 数据 修改 数据 以 及 
删除 数据 。 在 添加 数据 部 分 ， 着 重 讲解 了 如 何 给 自 增长 字段 添加 值 、 复 制 表 中 的 数据 以 及 


<“ 了 
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批量 添加 数据 ; 在 修改 数据 部 分 ， 着 重 讲解 了 如 何 修 改 前 N 条 记录 以 及 如 何在 修改 时 使 用 
其 他 表 中 的 数据 ;在 删除 数据 部 分 ， 着 重 讲解 了 如 何 删除 前 N 条 记录 ， 同 时 也 讲解 了 
TRUNCATE TABLE 语句 与 DELETE 语句 的 区 别 。 希 望 读者 通过 本 章 的 学 习 ， 能 够 熟练 地 


使 用 SQL 语句 来 操作 数据 表 中 的 数据 。 当 然 ， 企 业 管理 器 也 得 会 用 。 


5.6 本 章 习 题 


一 、 填 空 题 


1. 向 表 中 添加 数据 使 用 的 关键 字 是 
2. 修改 表 中 数据 使 用 的 关键 字 是 
3. 删除 表 中 数据 使 用 的 关键 字 是 
二 、 选 择 题 
1. 下 面 对 向 数据 表 中 插入 数据 的 描述 正确 的 是 
A. 可 以 一 次 向 表 中 的 所 有 字段 插入 数据 
B. 可 以 根据 条 件 向 表 中 的 字段 插入 数据 
C. 可 以 一 次 向 表 中 插入 多 条 数据 
D. 以 上 都 对 
2. 下 面 对 修改 数据 表 中 的 数据 描述 正确 的 是 
A. 一 次 只 能 修改 表 中 的 一 条 数据 B. 可 以 指定 修改 前 N 条 数据 
C. 不 能 修改 主键 字段 D. 以 上 都 不 对 
3. 下 面 对 删 除数 据 表 中 的 数据 描述 正确 的 是 
A. 使 用 DELETE 只 能 删除 表 中 的 全 部 数据 
B. 使 用 DELETE 可 以 删除 1 条 或 多 条 数据 
C. 使 用 TRUNCATE TABLE 语句 也 能 删除 1 条 或 多 条 数据 
D. 以 上 都 不 对 
三 、 问 答题 
1. INSERT 语句 的 基本 语法 形式 是 什么 ? 
2. UPDATE 语句 的 基本 语法 形式 是 什么 ? 
3. DELETE 与 TRUNCATE TABLE 的 区 别 是 什么 ? 
四 、 操 作 题 


使 用 表 5-1 所 示 的 游戏 账号 信息 表 ， 完 成 如 下 SQL 语句 的 编写 。 
(1) 向 游戏 账号 信息 表 中 任意 添加 5 条 数据 。 

(2) 将 游戏 账号 信息 表 中 前 3 条 记录 的 游戏 等 级 加 1。 

(3) 删除 所 有 游戏 等 级 为 1 的 游戏 账号 信息 。 


第 6 章 查询 语句 入 门 


查询 操作 可 谓 是 数据 表 中 最 重要 的 一 个 操作 了 。 如 果 没 ge 看 句 ， 那 么 数据 表 中 的 
数据 有 什么 变化 都 不 知道 。 这 就 好 像 有 一 个 仓库 只 能 存 取 东西 ， 但 是 就 是 不 知道 里 面 有 多 
少 东西 一 样 。 知道 它 的 重要 性 了 吧 ， 其 实在 上 - Bt Naan 
据 ， 那 就 是 SELECT 语句 。 在 本 章 中 将 继续 学 习 SELECT 查询 语句 更 多 的 用 法 。 

本 章 的 主要 知识 点 如 下 : 

口 运算 符 的 使 用 

口 如 何 书写 简单 的 查询 语句 

口 如 何在 查询 语句 中 使 用 聚合 函数 


6.1 运 算 符 


运算 符 这 个 词 读者 应 该 不 陌生 吧 ， 在 学 习 数学 或 者 是 编程 语言 时 ， 都 学 习 过 运算 符 。 
在 SQL Server 数据 库 中 ， 也 可 以 使 用 在 SQL 语句 中 使 用 运算 符 ， 特 别 是 查询 语句 中 。 有 
了 运算 符 ， 就 可 以 对 数据 表 中 的 数据 做 一 些 常用 的 统计 、 比 较 等 操作 了 。 因 此 ， 运 算 符 是 
很 重要 的 哦 。 在 SQL Server 数据 库 中 ,运算 符 主要 包括 算术 运算 符 、 比 较 运 算 符 、 迪 辑 运 
算 符 以 及 位 运算 符 等 。 下 面 就 将 逐一 讲解 每 一 类 运算 符 的 使 用 。 


6.1.1 算术 运算 符 


所 谓 算术 运算 符 ， 读 者 肯定 会 想到 就 是 用 来 进行 数学 运算 的 呐 。 没 错 ， 就 是 你 想到 的 
这 些 运 算 符 。 它 主要 包括 加 法 、 减 法 、 乘 法 、 除 法 、 取 余数 和 取 商 等 运算 符 。 有 具体 运算 符 
的 使 用 方法 如 表 6-1 所 示 。 


表 6-1 算术 运算 符 
运算 符 说 明 
对 两 个 操作 数 进行 加 法 运算 ,如 果 是 两 个 字符 串 类 型 的 操作 数 ， 则 可 以 将 两 个 字符 串 连 


站 接 到 一 起 。 比 如 : ahb=ab' 
三 对 两 个 操作 数 进行 减法 运算 
对 两 个 操作 数 进行 乘法 运算 
/ 对 两 个 操作 数 进行 除法 运算 ， 返 回 商 ， 例 如 :10/3-3 
% 对 两 个 操作 数 进行 取 余 运算 ， 返 回 余数 ， 例 如 ，10%3=1 


现在 就 用 示例 来 具体 地 解读 每 种 运算 符 的 具体 使 用 方法 。 在 讲解 示例 之 前 ， 先 要 告诉 
读者 SELECT 语句 不 仅 可 以 在 查询 数据 表 数 据 时 使 用 ， 也 可 以 直接 使 用 ， 相 当 于 赋值 或 者 
是 运算 使 用 。 在 下 面 的 示例 中 一 律 使 用 SELECT 语句 直接 通过 算术 运算 符 运 算 ， 不 使 用 数 


第 2 篇， 表 操作 基 础 


据 表 ， 以 便 读 者 更 好 地 理解 运算 符 的 使 用 。 
【示例 1】 使 用 “+”“--” 运 算 符 ， 计算 100 与 200、0.6 与 1.9 的 和 与 差 。 
根据 题目 的 要 求 ， 使 用 “+”“--” 运 算 符 运算 的 语句 如 下 : 


SELECT 100+200, 0.6+1.9, 100-200, 0.6-1.9; 


运算 结果 如 图 6.1 所 示 。 
读者 可 以 看 到 在 使 用 SELECT 语句 时 ， 并 没有 指定 要 使 用 的 数据 库 而 是 默认 地 用 了 
master 数据 库 。 实 际 上 ， 如 果 不 直 接 操作 数据 表 是 不 需要 指定 数据 库 的 ! 
【示例 2】 使 用 “* ”运算 符 ， 计 算 100 与 200、1.6 与 0.2 的 积 。 
根据 题目 的 要 求 ， 使 用 “* ”运算 符 运算 的 语句 如 下 : 


SELECT 100*200, 1.6*0.2; 


运算 结果 如 图 6.2 所 示 。 
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图 6.1 加 、 减 法 运算 符 的 使 用 图 6.2 乘法 运算 符 的 使 用 
【示例 3】 使 用 “/”“%” 运 算 符 ， 计 算 500 与 100、300 与 200、1.5 与 0.2 的 商 和 
运用 “/”“%” 运 算 符 计算 的 语句 如 下 
SELECT 500/100,500%100,300/200,300%200, 1.5/0.2, 1.5%0.2; 
运算 结果 如 图 6.3 所 示 。 
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图 6.3 ”除法 和 取 余 运算 符 的 使 用 


从 图 6.3 的 结果 中 可 以 看 出 ， 如 果 两 个 数 能 够 整除 时 ， 余 数 就 为 0。 即 使 是 小 数 之 间 
做 取 余 运算 也 是 得 到 余数 ， 并 且 余 数 也 是 小 数 。 比 如 : 1.5%0.2=0.1。 
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6.1.2 ”比较 运算 符 

比较 运算 符 是 用 来 对 两 个 操作 数 进行 比较 的 。 读 者 可 以 想 想 ， 两 个 操作 数 之 间 比 较 结 
果 应 该 是 什么 呢 ? 通过 比较 运算 符 运算 的 结果 不 是 具体 的 数值 ， 而 是 用 布尔 类 型 表示 的 ， 
即 TRUE 或 FALSE。 在 SQL Server 数据 库 中 ， 比 较 运算 符 主 要 有 大 于 、 小 于 、 大 于 等 于 、 
小 于 等 于 和 不 等 于 等 ， 具 体 的 使 用 方法 如 表 6-2 所 示 。 
表 6-2 ”比较 运算 符 


明 运 


在 表 6-2 所 列 出 的 运算 符 中 ， 表 示 不 等 于 的 有 两 种 方式 :“<>” 和 “!=”， 它们 在 使 用 
上 没有 任何 区 别 ， 就 根据 你 自己 的 喜好 来 选择 吧 。 这 里 需要 注意 的 是 ， 在 SELECT 语句 后 
面 不 能 够 直接 使 用 比较 运算 符 对 值 进行 比较 。 比 较 运 算 符 通常 会 用 在 查询 语句 中 的 
WHERE 子 名 或 者 是 工 SQL 编程 时 的 语句 中 。 因 此 ， 比 较 运 算 符 的 具体 使 用 方法 将 在 本 章 
的 查询 部 分 详细 讲解 。 


6.1.3 ”逻辑 运算 符 


好 和 辑 运算 符 与 比较 运算 符 有 些 相 似 ， 运 算 的 结果 都 是 布尔 类 型 的 。 光 辑 运 算 符 也 主要 
应 用 于 查询 语句 的 WHERE 子 句 中 。 在 SQL Server 数据 库 中 ,好 辑 运算 符 主 要 有 and、or、 
not、all 和 in 等 。 风 辑 运 算 符 的 具体 使 用 方法 如 表 6-3 所 示 。 


表 6-3 ”逻辑 运算 符 


运算 符 说 明 

Pm 与 运算 符 。 用 于 两 个 操作 数 比 较 时 ， 当 两 个 操作 数 同时 为 TRUE 时 ， 结 果 才 是 TRUE; 
否则 是 FALSE 

和 或 运算 符 。 用 于 两 个 操作 数 比 较 时 ， 当 两 个 操作 数 同时 为 FALSE 时 ,结果 才 是 FALSE; 
否则 是 TRUE 

非 运算 符 。 对 任意 运算 结果 的 布尔 类 型 值 取 反 。 即 表达 式 的 结果 为 TRUR 时 , 通过 NOT 
运算 符 运算 后 ， 结 果 为 FALSE; 否则 为 TRUE 

pe 用 于 判断 是 否 都 满足 条 件 。 比 如 : 100>ALL(10,30,50,60)，100 要 大 于 ALL 后 面 的 每 一 
个 值 时 ， 结 果 是 TRUE， 否 则 为 FALSE 

Pe 于 判断 是 否 有 一 个 值 满足 条 件 。 比 如 : 100>ANY(40,101)， 只 要 在 ANY 后 面 有 一 个 数 
是 大 于 100 的 ， 结 果 就 是 TRUE， 否则 就 是 FALSE 

SOME 与 ANY 的 使 用 方法 相同 

页 判断 某 一 个 值 是 否 在 IN 后 面 的 指定 范围 内 。 比 如 : 100 IN (100，200，300)， 如 果 在 IN 
后 面 的 数值 中 有 100， 那 么 结果 为 TRUE， 和 否则 为 FALSE 
判断 某 一 个 值 是 否 在 一 个 范围 内 。 通 常情 况 下 ，BETWEEN 与 AND 连用 ， 表 示 一 个 有 具 

BETWEEN | 体 的 范围 。 比如 : 100 BETWEEN 50 AND 200, 如 果 100 在 50 一 200 之 间 , 结果 是 TRUE， 
否则 是 FALSE 

EXISTS 判断 是 否 能 查询 出 数据 。 通 常 在 子 查询 中 使 用 
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关于 逻辑 运算 符 的 具体 使 用 方法 也 将 在 后 面 的 查询 中 详细 讲解 。 这 里 ， 只 需要 读者 先 
了 解 一 下 逻辑 运算 符 的 表示 方法 。 
6.1.4 位 运算 符 

位 运算 符 实际 上 就 相当 于 是 对 数值 的 运算 ， 因 此 ， 也 可 以 在 SELECT 语句 后 面 直接 使 
j。 在 SQL Server 数据 库 中 ， 位 运算 符 有 按 位 与 、 按 位 或 和 按 位 异 或 3 种 。 位 运算 符 的 具 
体 使 用 方法 如 表 6-4 所 示 。 


表 6-4 位 运算 符 
说 了 明 

按 位 与 ， 是 将 两 个 操作 数 转换 成 二 进 制 数 ， 然 后 按 位 进行 比较 ， 如 果 比 较 的 对 应 ， 两 位 
同时 为 1 则 结果 为 1， 否 则 全 都 是 0 
按 位 或 ， 是 将 两 个 操作 数 转换 成 二 进 制 数 ， 然 后 按 位 进行 比较 ， 如 果 比 较 的 对 应 两 位 同 
时 为 0 则 结果 为 0， 和 否则 全 都 是 1 
按 位 异 或 ， 是 将 两 个 操作 数 转 换 成 二 进 制 数 ， 然 后 按 位 进行 比较 ， 如 果 比 较 的 对 应 两 位 
的 值 相同 则 结果 为 0， 否 则 是 


由 于 位 运算 符 可 以 直接 应 用 在 SELECT 语句 后 面 ， 那 下 面 就 让 我 们 先 一 睹 为 快 吧 。 
【示例 4】 使 用 按 位 与 运算 符 计算 5 与 2， 使 用 按 位 或 运算 符 计算 10 与 8， 使 用 按 位 
异 或 运算 符 计 算 10 与 6 的 结果 。 
使 用 位 运算 符 运算 的 语句 如 下 : 
SELECT 5&2, 10|8, 10^6; 
也 6 可 


运算 结果 如 图 6.4 所 示 。 nr 
看 了 图 6.4 的 运算 结果 ， 读 者 有 些 糊涂 了 吧 ， 怎 么 能 得 “| 政 晶 法 易 现 钾 剖 曙 
出 这 样 的 结果 。 现 在 就 给 你 分 析 一 下 吧 ， 先 要 把 位 运算 的 操 assawj 已 已 GD 
作 数 转换 成 二 进 制 数 ， 然 后 再 运算 。 比 如 : 5 的 二 进 制 表示 理 如 ee 
是 00000101，2 的 二 进 制 表示 是 00000010， 那么 ， 按 位 或 的 
结果 是 00000000。 因 此 ， 结 果 就 是 0 了。 其 他 的 操作 数 在 这 
里 就 不 一 一 转换 了 ， 请 读者 自己 转换 一 下 ， 然 后 再 对 比 一 下 | 
运算 结果 。 s ，| master | 00:00:00 | 1 行 
19 ch 地 


6.1.5 其 他 运算 符 图 6.4 ”位 运算 符 的 使 
除了 上 面 的 4 类 运算 符 之 外 , 还 有 一 元 运算 符 、 赋 值 运算 符 等 。 赋值 运算 符 就 是 等 号 ， 
让 “=” 来 表示 ， 这 里 就 不 多 说 了 。 所 谓 一 元 运算 符 ， 就 是 对 一 个 表达 式 或 操作 数 进行 运 
算 的 ， 具 体 的 使 用 方法 如 表 6-5 所 示 。 


表 6-5 ”一 元 运算 符 
运 算 符 说 了 明 
+ | 表示 操作 数 的 值 是 正 值 
二 | 表示 操作 数 的 值 是 负 值 


返回 表达 式 的 结果 或 操作 数 的 补 数 ， 也 称 取 反 运算 。 该 运算 通常 用 于 二 进 制 数 
的 取 反 运算 ， 将 一 个 二 进 制 数 的 每 一 位 ， 是 0 的 换 成 1， 是 1 的 换 成 0 


a 
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一 元 运算 符 也 是 可 以 在 SELECT 语句 后 面 直接 使 用 的 ,“+” 和 “-” 运 算 符 比较 简单 ， 
只 是 给 操作 数 前 面 加 上 一 个 符号 而 已 。 下 面 就 重点 学 习 取 反 运 算 符 的 使 用 。 

【示例 5】 使 用 一 元 运算 符 中 的 取 反 运算 符 给 10 取 反 。 

使 用 一 运算 符 对 10 取 反 的 语句 如 下 : 


SELECT ~10; 


运算 结果 如 图 6.5 所 示 。 
从 图 6.5 的 运算 结果 中 , 读者 又 迷糊 了 吧 。 那 咱 们 还 是 来 “EROETCC 
先 把 10 转换 成 二 进 制 数 看 看 ，10 的 二 进 制 是 00000000 ， 钼 日 开 2 
00001010，~10 的 结果 是 11111111 11110101， 这 样 第 1 位 是 ”| 让 | 怨 钙 名店 量 
1 表示 是 负数 ， 那 么 ， 取 补 码 就 应 该 是 10000000 00001011， FE 
转换 成 十 进 制 数 结果 就 是 -11 了 。 本 本 
器 结果 | 消息 | 

全 说 明 : 补 码 的 取 值 方法 是 ， 对 于 所 有 的 有 符号 整数 ， 最 高 | 

位 1 的 表示 负数 ， 最 高 位 0 的 表示 正 数 。 一 般 来 讲 ， jz7Wdninistrate ,，| master |00:00:00 |1 行 
有 符号 正 整 数 按 位 取 反 后 就 是 其 相反 数 减 1， 负 数 {mm 
取 反 就 是 相反 数 加 1。 图 6.5 取 反 运算 符 的 使 用 


6.1.6 运算 符 也 是 有 顺序 的 
读者 学 习 到 这 里 ， 就 意味 着 运算 符 已 经 学 完了 。 虽 然 运 算 符 学 完了 ， 但 是 如 果 要 在 
-个 表达 式 中 使 用 多 个 运算 符 时 ， 运 算 符 之 间 也 是 有 顺序 的 。 这 就 像 去 银行 办 理 业务 
如 果 你 是 VIP 客户 ， 那 么 就 会 优先 的 。 下 面 就 用 表 6-6 来 说 明 运算 符 的 优先 级 是 什么 
样 的 。 
表 6-6 ”运算 符 的 优先 级 


顺 序 运 算 符 
1 ~ 
2 .se 
3 +、 一 、 坟 
于 = (判断 相等 )、>、<、>=、<=、!=、!>、< 
5 二 让 
6 NOT 
7 AND 
8 ALL、 SOME ANY、 BETWEEN, OR 
9 二 (赋值 》 


从 表 6-6 所 示 的 运算 符 的 优先 级 ， 可 以 看 出 最 后 执行 的 是 赋值 运算 符 、 最 先 执行 的 是 
取 反 运算 符 。 在 实际 的 应 用 中 ， 也 不 必 总 是 考虑 每 个 表达 式 中 要 先 执行 什么 后 执行 什么 ， 
只 需要 把 先 执行 的 内 容 用 括号 括 起 来 ， 这 样 就 可 以 避免 一 些 不 必要 的 麻烦 了 。 
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6.2 简单 查询 


在 本 节 中 介绍 的 简单 查询 是 指 对 一 张 表 的 查询 操作 。 查 询 操作 是 对 表 操作 的 一 个 重要 
操作 ， 使 用 的 关键 字 是 SELECT。 读 者 对 于 这 个 关键 字 可 能 不 陌生 了 ， 但 是 真正 使 用 好 查 
询 还 真 不 是 一 件 容易 的 事 啊 。 因 此 ， 打 好 基础 是 关键 ， 只 有 掌握 了 简单 查询 才能 更 好 地 掌 
握 下 一 章 中 的 复杂 查询 。 准 备 好 了 吗 ? 让 我 们 一 起 进入 查询 语句 的 世界 吧 。 


6.2.1 查询 语句 的 基本 语法 形式 


任何 一 种 语句 都 有 特定 的 语法 形式 , 查询 语句 也 不 例外 。 查询 语句 的 语法 形式 很 复杂 ， 
在 这 里 先 给 出 一 个 比较 容易 理解 的 形式 ， 对 于 其 他 的 形式 将 在 后 面 的 内 容 中 继续 讲解 。 查 
询 语句 的 语法 形式 如 下 : 
SELECT column namel, column name2, column name3. 
FROM table name 
WHERE conditions; 
其 中 : 
口 column_namel: 列 名 。 要 在 查询 结果 中 显示 的 列 信 息 ， 多 个 列 之 间 用 逗号 隔 开 。 
如 果 在 查询 中 想 查 看 表 中 的 全 部 信息 ， 列 名 就 不 用 一 一 列 出 了 ， 直 接 用 “*” 号 代 
替 就 可 以 了 。 
口 table_name: 表 名 。 要 查询 数据 的 表 名 。 在 查询 前 也 要 先 打开 该 表 所 使 用 的 数据 库 。 
口 conditions: 条 件 。 按 什么 条 件 来 查询 数据 。 条 件 可 以 是 1 到 多 个 ， 多 个 条 件 之 间 
就 会 用 到 前 面 我 们 学 过 的 运算 符 了 ， 最 常用 的 是 多 辑 运算 符 。 
6.2.2 ”把 表 里 的 数据 都 查 出 来 
从 数据 表 中 查询 全 部 数据 ， 这 个 操作 读者 肯定 用 到 过 。 在 上 一 章 对 表 的 操作 中 ， 经 常 
会 使 用 这 种 方法 来 验证 操作 结果 。 如 果 你 没 记 住 ， 也 不 要 紧 ， 现 在 跟着 示例 6 再 演练 一 下 
就 明白 了 。 在 演练 之 前 ， 先 说 明 一 下 本 章 中 使 用 的 数据 库 和 数据 表 。 数 据 库 使 用 chapter6 。 
数据 表 使 用 小 说 信息 表 , 统计 小 说 的 信息 应 该 有 什么 呢 ? 一 部 小 说 , 通常 要 知道 它 的 名 字 、 
作者 、 类 型 和 点 击 率 等 信息 。 具 体 的 表 结 构 如 表 6-7 所 示 。 


表 6-7 ”小 说 信息 表 (novelinfo) 


编号 列 名 数据 类 型 中 文 释义 
1 id INT 编号 
2 name VARCHAR(20) 名 称 
3 author VARCHAR(20) 作者 
4 noveltype VARCHAR(20) 类 型 (网 络 版 、 纸 版 、 手 机 版 》 
6 CTR INT 点 击 率 
7 | remar 备注 
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由 于 本 章 只 是 对 数据 表 中 的 内 容 做 查询 操作 ， 因 此 ， 还 要 先 为 表 中 输入 一 些 数据 。 表 
中 输入 的 数据 如 表 6-8 所 示 。 


表 6-8 小 说 信息 表 中 的 数据 


名 称 
1 | 斗 破 苍 容 100 | 无 
2 “| 斗 罗 大 陆 120 | 无 
3 | 极品 公子 100 | 无 
4 ”| 101 条 花王 狗 200 | 中 国 对 外 翻译 出 版 公司 
5 10 号 公寓 国 华 侨 出 版 社 


创建 小 说 信息 表 并 录入 数据 的 语句 如 下 : 


USE chapter6; 
CREATE TABLE novelinfo 
( 
id INT IDENTITY(1,1) PRIMARY KEY, 
name VARCHAR(20), 
author VARCHAR(20), 
noveltype VARCHAR(20), 
role VARCHAR(20), 
CTR INT, 
remark VARCHAR(200) 


Wy 
INSERT INTO novelinfo VALUES (' 斗 破 苍 穹 ',' 天 佛 土豆 '，' 网 络 版 ', ' 萧 炎 ' ,100, ' 无 ')， 
(' 斗 罗 大 陆 '，' 唐 家 三 少 ',' 网 络 版 ',' 唐 三 ' ,120, ' 无 ')， 
(' 极 品 公子 ',' 烽 火 戏 诸侯 '，' 网 络 版 ',' 叶 无 道 ' ,100, ' 无 ')， 
('101 条 花 斑 狗 ',' 史密斯 ',' 纸 版 ',' NULL' ,200,， 中 国 对 外 翻译 出 版 公司 ')， 
('10 号 公寓 ',' 诸 葛 宇 聪 ',' 纸 版 ', 'NULL' ,300,， 中 国 华侨 出 版 社 ') 
执行 上 面 的 语句 ,就 可 以 完成 小 说 信息 表 的 创建 。 下 面 就 该 使 用 SQL 语句 来 查询 小 说 
信息 表 了 。 
【示例 6】 查询 小 说 信息 表 (novelinfo) 的 全 部 数据 。 
根据 题目 的 要 求 ， 要 查询 小 说 信息 表 中 的 全 部 数据 。 那么 , 并 不 需要 知道 表 中 的 列 名 ， 
只 需要 使 用 “*” 代 替 列 名 就 可 以 了 。 查 询 语 句 如 下 : 
USE chapter6; 
SELECT * FROM novelinfo; 
执行 上 面 的 语句 ， 就 可 以 查看 表 中 的 全 部 数据 了 。 执 行 效果 如 图 6.6 所 示 。 
读者 可 以 将 图 6.6 的 查询 结果 与 表 6-8 所 示 的 数据 对 比 一 下 ， 对 比 后 就 会 发 现 原 来 查 
询 表 中 的 全 部 记录 就 是 如 此 简单 的 啊 。 
6.2.3 ”查看 想 要 的 数据 
虽然 查询 表 中 的 全 部 记录 是 很 简单 的 事情 ， 那 么 ， 每 次 都 是 要 查询 表 中 的 数据 为 何 还 
要 学 习 其 他 的 查询 语句 呢 ? 原因 很 简单 ， 虽 然 衣 柜 里 有 很 多 衣服 ， 但 并 不 是 每 次 洗衣 服 时 
都 要 全 部 拿 出 来 洗 一 次 ， 都 是 只 洗 脏 衣服 的 。 数 据 表 也 是 一 样 ， 通 常情 况 下 ， 并 不 需要 每 
次 都 查询 表 中 的 全 部 记录 。 另 外 ,每 次 都 从 表 中 查询 全 部 数据 ， 也 会 影响 查询 效率 。 因 此 ， 
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了 microesoft SQL Server Nanagenent Studio =Iolxl 
文件 到 ) 编辑 于 ) 视图 WD) 查询 @) 项 目 E) 调式 @W) 工具 CD) 窗口 DD 
社区 CC) 帮助 

卫 新建 查询 中 疝 | 六 访 钙 | 启 | 蕊 回 马 总: 

台 8 | chptere -| 》 mv 时 名 国 | 由 本 
SqLQueryl. s... ator (53))*| -x 


USE chapter6; 
| emer + 了 RON novelinfo; 


司 结果 | 六 请 息 | 


1 | 斗 破 苍 容 天 要 土 豆 网 络 版 ” 利 伙 100 

| 天 十 2 斗 罗 大 陆 。” 唐 家 = 少 。 网 络 版 ” 唐 120 

| 轩 | 3 极品 公子 。 ”烽火 戏 诸侯 ”网 络 版 ” 叶 无 道 100 无 

| | 4 10 条 花王 洛 ”史密斯 纸 版 NULL 。 200 ”中 国 对 外 翻译 出 版 公司 
5 10 号 公寓 诸 夏 宁 陪 纸 版 NULL ”300 ”中 国 华侨 出 版 社 


| WIDY-9527 MSSQLSERVER2008 〔 . ，| WIDW-9527\Adninistrato. .. | chapter6 | 00:00:00 | 5 行 
就 结 行 4 列 1 Ch 1 | 


图 6.6 查询 小 说 信息 表 中 的 全 部 数据 


学 习 如 何 查 询 想 要 的 数据 还 是 很 重要 的 。 下 面 就 通过 示例 7 来 演示 一 下 。 
【示例 7】 查询 小 说 信息 表 (novelinfo) 表 中 的 小 说 名 称 (name) 和 小 说 作者 (author)。 
在 查询 之 前 ， 先 要 明确 一 下 查询 的 语法 规则 ， 要 查询 出 哪个 列 的 信息 就 在 SELECT 语 
句 中 指定 该 列 名 。 具 体 的 查询 语句 如 下 : 
USE chapter6; 
SELECT name author FROM novelinfo; 
执行 上 面 的 语句 ， 就 可 以 查询 到 指定 列 名 的 相关 数据 了 。 执 行 效果 如 图 6.7 所 示 。 
从 图 6.7 中 可 以 看 出 ， 显 示 的 结果 中 只 含有 
SELECT 语句 后 面 指定 的 小 说 名 称 和 小 说 作者 的 数 。 四 CED rE 


据 了 工具 WD 窗口 WD 社区 C) 帮助 由 
及 8D | 访 | 翅 迎 久 | 己 | 区 器 马 | 加 
By 2 | ceptere -| 的) >》 六 


6.2.4 给 查询 结果 中 的 列 换个 名 称 


通过 前 面 的 示例 6 和 示例 7, 读者 已 经 对 查询 语 
句 有 所 了 解 了 。 在 查询 的 结果 中 ， 看 到 的 列 标题 就 
是 数据 表 中 的 列 名 。 如 果 不 是 表 的 设计 者 ， 有 时 真 
的 很 难 知道 字段 的 意义 啊 。 那 么 ， 如 何 能 够 按照 自 
己 的 意愿 去 定义 列 名 呢 ? 在 SQL Server 中 ， 可 以 使 


.| TDY-sszTWAdninistrate .. | chapter8 | 00:00:00 |5 行 


用 给 列 定义 别名 的 方法 来 完成 。 并 且 ， 在 给 列 定 义 和 1 Wi 
别名 时 ， 有 如 下 3 种 方法 。 下 面 就 带 你 认识 一 下 吧 。 图 6.7 查询 指定 列 的 数据 
(1) 使 用 AS 关键 字 给 列 设置 别名 


SELECT column_namel RS ' 别 名 1'，column _name2 AS ' 别 名 2',， column name3 RS " 
FROM table name 


其 中 ， 在 AS 后 面 的 名 字 就 是 给 其 前 面 的 列 定义 的 别名 。 
(2) 使 用 空格 给 列 设置 别名 


SELECT column_namel ' 别 名 1', column name2 ' 别 名 2', column name3 ' 别 名 3'.... 
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FROM table name 


其 中 ， 空 格 后 面 的 名 字 就 是 给 其 前 面 的 列 设置 的 别名 。 
(3) 使 用 等 号 给 列 设置 别名 


SELECT ' 别 名 1'=column namel，“' 别 名 2'=column namel，' 别 名 3'=column 


FROM table name 


其 中 ， 等 号 前 面 的 名 字 就 是 给 其 后 面 的 列 设置 的 别名 。 

读者 可 以 想 想 ， 上 面 的 3 种 方式 中 哪 种 是 最 常用 的 呢 ? 实 际 上 ， 第 一 种 方法 是 最 常用 
的 ， 不 仅 可 以 让 列 名 和 别名 之 间 有 个 区 分 ， 又 可 以 让 其 看 起 来 更 舒服 些 。 虽 然 第 一 种 方法 
是 最 常用 的 ， 但 是 其 他 的 方法 也 要 掌握 哦 。 

下 面 就 用 示例 8 来 演示 一 下 如 何 运用 这 3 种 方法 给 列 设置 别名 。 

【示例 8】 查询 小 说 信息 表 ， 显 示 小 说 的 名 称 、 作 者 名 称 以 及 点 击 率 信息 。 并 分 别 使 
用 上 面 的 3 种 的 方式 为 列 设置 别名 。 

根据 题目 的 要 求 ， 只 需要 在 SELECT 后 面 指定 3 个 列 名 。 查 询 语句 如 下 : 

(1) 使 用 AS 关键 字 设置 别名 

USE chapter6; 


SELECT name AS ' 名 称 ', author RS ' 作 者 ' ,CTR RS ' 点 击 率 ' 
FROM novelinfo; 


执行 上 面 的 语句 , 就 可 以 将 SELECT 后 面 指定 列 的 内 容 查 询 出 来 了 。 执行 效果 如 图 6.8 
所 示 。 


Ricrosoft SQL Server Nanagement Stodio =|Djxl 


文件 人 ”编辑 亿 ) 视图 查询 @) 项 目 @) 调试 Q) 工具 GD) 窗口 四 
社区 加 帮助 中 


明光 WD | 让 | 也 也 吕 | 记 | 芝 加 乌 | 更 日 
Ry Wa | chaptere -| 执行) 》 mv 踢 邓 加 | 3 
SQLQueryl. s... ator (53))*| 2 


lh-9527\SSQLSERYER2008 〔 .，| YITDY-9527\Adninistrato .| chapter8 |00:00:01 |5 行 
行 6 列 1 ED 田 


图 6.8 ”使 用 AS 关键 字 给 列 设置 别名 


(2) 使 用 空格 给 列 设置 别名 


USE chapter6; 
SELECT name ' 名 称 ',author ' 作 者 ', CTR ' 点 击 率 ' 
FROM novelinfo; 


执行 上 面 的 语句 ， 也 可 以 完成 与 (1) 同样 的 效果 。 
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(3) 使 用 等 号 给 列 设置 别名 


USE chapter6; 
SELECT ' 名 称 '=name，' 作 者 '=author，' 点 击 率 '=CTR 
FROM novelinfo; 


执行 上 面 的 语句 ， 也 可 以 完成 与 (1) 同样 的 效果 。 
6.2.5 使 用 TOP 查询 表 中 的 前 几 行 数据 


在 前 面 的 查询 中 ， 虽 然 可 以 指定 要 显示 的 列 ， 但 是 查询 的 数据 也 是 表 中 的 数据 。 每 次 
都 要 显示 表 中 的 所 有 数据 ， 如 果 数 据 很 少 的 时 候 速 度 还 不 会 受到 影响 ， 如 果 数 据 量 很 大 时 
就 会 造成 查询 时 间 过 长 、 数 据 库 访 问 速度 降低 的 情况 。 当 然 ,在 SQL Server 中 会 有 办 法 解 
决 这 个 问题 的 , 那 就 是 使 用 TOP 关键 字 来 限制 显示 结果 的 数量 。TOP 关键 字 可 以 帮助 读者 
每 次 仅 返回 查询 结果 的 前 N 行 。 具 体 的 使 用 方法 请 读者 认真 学 习 示例 9。 

【示例 9】 查询 小 说 信息 表 中 的 前 2 条 记录 。 显 示 出 小 说 的 名 称 以 及 小 说 类 型 的 信息 。 

根据 题目 要 求 ， 在 SELECT 语句 后 面 列 出 小 说 名 称 和 小 说 类 型 的 字段 就 可 以 了 ， 并 在 
前 面 加 上 TOP(2)。 具 体 的 语句 如 下 : 

USE chapter6; 

SELECT TOP(2) name as ' 名 称 ', noveltype as ' 类 型 ' FROM novelinfo; 


执行 上 面 的 语句 ， 就 可 以 查询 出 小 说 信息 表 的 指定 列 的 前 2 条 记录 了 。 执 行 效果 如 
图 6.9 所 示 。 


Microsoft SQL Server Nanagenent Studio =|Djxl 
文件 @) 编辑 下 ) 视图 查询 @) 项目) 调 坛 @) 工具 GD) 窗口 WD 社区 C) 

帮助 0) 

了 .新建 查询 叫 | 丹 | 启 启 驯 | 市 | 蕊 回 对 加 局 

BY 2 | chaptere "| 执行) 有 v 中 本国 | 3 支 图 的 
SQLQueryl1. s... ator (54))*| vx 


日 0SE chapter6; 


| SELECT TOP (2) name as ' 名 称 ' ,noveltype as ' 类 型 ，FROM novelinfo; 


.| TDY-9527\Adninistrate .. | chapter8 | 00:00:00 | 2 行 
行 5 列 1 Chl Ins /4 


图 6.9 使 用 TOP 查询 出 表 中 的 前 2 条 记录 
从 图 6.9 的 查询 结果 中 可 以 看 出 ， 确 实 是 只 查询 了 结果 中 的 前 两 条 记录 了 。 
6.2.6 在 查询 时 去 除 重复 的 结果 


在 查询 操作 时 有 时 希望 去 除 一 些 重复 的 数据 ， 以 便 查 看 数据 。 其 实 这 个 并 不 难 实现 ， 
只 需要 在 SELECT 语句 后 面 加 上 DISTINCT 关键 字 就 可 以 了 。 实 际 上 ， 在 真正 的 数据 表 中 
两 条 完全 重复 的 记录 是 很 少见 ， 只 是 某 一 列 或 多 列 重 复 的 记录 会 多 一 些 。 无 论 是 去 除 表 中 
重复 的 记录 ， 还 是 去 除 某 列 的 重复 值 ，DISTINCT 关键 字 都 可 以 帮 你 搞定 。 
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【示例 10】 查询 小 说 信息 表 ， 显 示 小 说 的 类 型 ， 并 去 除 重 复 的 信息 。 

根据 题目 要 求 ,在 SELECT 语句 后 面 加 上 小 说 类 型 的 字段 ,并 在 其 前 面 加 上 DISTINCT 
关键 即 可 。 查 询 语句 如 下 : 

USE chapter6; 

SELECT DISTINCT noveltype FROM novelinfo; 

执行 上 面 的 语句 ， 即 可 查看 到 小 说 信息 表 中 的 小 说 类 型 信息 了 。 执 行 效果 如 图 6.10 
所 示 。 

从 图 6.10 所 示 的 结果 中 可 以 看 出 , 在 小 说 信息 。。 交加 各 视 四 WD 考 洒 项目 中 总 


Ss y 工具 QD) 窗口 ) 社区 C) 帮助 0 
表 中 只 有 2 个 小 说 类 型 。 ET 


nd -| ?执行 o >》 而 各 
6.2.7 ”查询 结果 也 能 排序 i 


在 一 些 大 型 的 网 站 上 ， 读 者 经 常会 看 到 一 些 流 
行 音乐 排行 榜 、 人 气 排行 榜 等 等 信息 。 这 些 信息 也 
可 以 通过 从 数据 库 中 查询 数据 来 获得 。 读 者 可 以 想 
一 想 ， 在 查询 语句 中 什么 时 候 对 结果 进行 排序 呢 ? ee |eere [oom mm [2 
当然 是 在 查询 语句 的 最 后 面 对 其 查询 结果 进行 排 2 
序 了 。 对 查询 结果 进行 排序 要 使 用 ORDER BY 语 ”图 510 DISTINCT 关键 字 的 使 用 
句 来 完成 。 对 查询 结果 排序 的 语法 格式 如 下 所 示 。 


SELECT column namel, column name2, column name3..... 

FROM table name 

WHERE conditions 

ORDER BY column namel DESCIASC, column name2 DESCIASC.... 

这 里 ， 除 了 ORDER BY 语句 外 ， 其 他 语句 在 前 面 的 查询 语句 的 语法 中 已 经 解释 过 了 ， 
这 里 就 不 再 解释 了 。ORDER BY 子 句 后 面 可 以 放置 1 列 或 多 列 ， 在 每 一 列 后 面 还 要 指定 该 
列 的 排序 方式 ，DESC 代表 的 是 降序 排列 ，ASC 代表 的 是 升序 排列 。 如 果 不 对 数据 表 中 的 
数据 进行 排列 ， 默 认 的 排序 方式 是 升序 排列 。 

【示例 11】 查询 小 说 信息 表 ， 并 对 小 说 信息 表 中 的 点 击 率 进行 降序 排列 。 

根据 题目 要 求 ， 可 以 将 表 中 的 全 部 数据 查询 出 来 ， 然 后 再 使 用 ORDER BY 排序 。 具 
体 的 语句 如 下 : 

USE chapter6; 

SELECT * FROM novelinfo ORDER BY CTR DESC; 

执行 上 面 的 语句 , 就 可 以 将 查询 结果 按照 CTR 列 降序 排列 了 。 执行 效果 如 图 6.11 所 示 。 

从 图 6.11 的 结果 可 以 看 出 ， 查 询 结 果 确 实 是 按照 CTR 列 从 大 到 小 排列 的 。 


6.2.8 含有 NULL 值 的 列 也 能 查看 


在 前 面 的 查询 语句 中 都 是 查询 表 中 的 全 部 数据 ， 如 何 按 某 个 条 件 来 查询 数据 呢 ? 那 就 
要 用 到 查询 语句 中 的 WHERE 子 名 了。 在 WHERE 子 句 中 就 可 以 指定 按 什么 条 件 来 查询 数 
据 。 数 据 表 中 的 NULL 值 是 会 经 常 出 现 的 ，NULL 值 通常 是 指 没 有 录入 数据 的 列 ， 那 如 何 
查看 含有 NULL 值 的 列 呢 ? 请 看 下 面 WHERE 语句 第 一 个 应 用 吧 。 
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帮助 中 


也 新 于 查询 中 | 让 轧 也 可 上 巴 | 记 加 与 | 受 瑟 
于 Wa | chaptere -| ?执行 Oh 有 v 史 时 加 | 下 本 | 鸡 名 
“SQLQueryl. s... ator (54))#| vx 


日 USE chapter6; 
| SELECT * FROM novelinfo ORDER BY CTR DESC; 


10 号 公寓 诸 融 字 陪 ” 纸 版 NULL ”300 ”中 国 华侨 出 版 社 
4 ”101 条 花王 狗 “史密斯 。 纸 版 NULL ”200 ”中 国 对 外 翻译 出 版 公司 
2 斗 罗 大 陆 唐家三 少 ”了 9 络 版 唐 120 无 
3 
1 


eho rovelype [sole [cihlmsak | | 


极品 公子 烽火 戏 .、， 网 络 版 叶 100 无 
斗 破 苍 窜 天 要 土豆 ”网 络 版 ”第 炎 100 
@| ID-9527 SSQLSERVER2008 (.. | WIDW-9527\Adninistrato. . 
就 绪 行 4 列 1 


图 6.11 对 查询 结果 使 用 ORDER BY 排序 
【示例 12】 查询 小 说 信息 表 ， 并 显示 出 所 有 小 说 主角 是 NULL 值 的 数据 。 
根据 题目 要 求 , 在 查询 时 要 使 用 WHERE 子 句 来 限制 只 查询 出 小 说 主角 是 NULL 的 数 
据 。 具 体 的 语句 如 下 : 


USE chapter6; 
SELECT * FROM novelinfo WHERE role is NULL; 


执行 上 面 的 语句 ， 就 可 以 查 到 主角 为 NULL 的 数据 了 。 执 行 效果 如 图 6.12 所 示 。 


Ricrosoft SQL Server Nanagenent Studio 3 =Io|x| 
文件 @) 编辑 EE) 视图 WW 查询 @) 项 目 E) 调式 W) 工具 CD) 窗口 人 


.| chspters |00:00:00 |5 行 
Chl - 


社区 CC) 帮助 中 
也 新建 查询 | 六 | 售 启 态 | 店 | 区 加 对 | 凤 避 
;RY BR | chaptere "| ?执行 DO 》 提 中 名 国 


SQLQueryl. s. 
日 0SE chapt 
| SELECT * 


4)*| -x 


novelinfo WHERE role is NULL; 


[|_ jalname | ovelyp emak 
[4 | 101 条 花王 狗 。” 史 窑 斯 。 纸 版 。 ”NULL ”200 ”中 国 对 外 翻译 出 版 公 
| 加 加 5 ”10S 公 富 。 诸 融 字 陪 纸 版 。 NULL ”300 中 国 华侨 出 版 社 


eT\MSSQLSERVER2008 (.. | NIDN-9527\Adninistrato... | chapter6 |00:00:00 | 2 行 
就 绪 行 4 列 1 chl 及 


图 6.12 查询 含有 NULL 值 的 列 


这 里 ， 需 要 注意 一 下 查询 含有 NULL 值 的 列 ， 需 要 使 用 “ 列 名 is NULL” 的 语法 格式 
而 不 能 直接 使 用 “ 列 名 =NULL ”的 格式 。 


6.2.9 ”模糊 查询 用 LIKE 


所 谓 模糊 查询 ， 就 好 像 在 百度 中 搜索 东西 一 样 ， 输 入 一 个 词 或 一 句 话 就 会 输出 与 之 相 
关 的 内 容 。 在 数据 库 中 ， 模 糊 查询 是 通过 LIKE 关键 字 来 完成 的 。 但 是 ， 在 学 习 LIKE 关 
键 字 之 前 读者 先 要 记 住 几 个 通配符 ， 如 表 6-9 所 示 。 
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表 6-9 通配符 
运 算 符 说 明 
% 表示 0 到 多 个 字符 
_ 表示 一 个 单个 字符 

0 表示 含有 [内 指定 的 字符 


知道 了 表 6-8 中 的 通配符 ， 现 在 就 可 以 使 用 LIKE 进行 模糊 查询 了 。 如 果 想 表示 “不 
像 .….” 的 意思 ， 可 以 使 用 “NOT LIKE” 来 查询 。 

【示例 13】 查询 小 说 信息 表 ， 并 查询 出 小 说 名 字 中 含有 “10” 的 信息 。 

根据 题目 要 求 ， 小 说 名 字 含 有 “10” 就 意味 着 这 个 “10” 可 以 在 小 说 名 字 的 开头 、 中 
间 以 及 结尾 中 任意 位 置 出 现 。 具 体 的 查询 语句 如 下 : 


USE chapter6; 
SELECT * FROM novelinfo WHERE name LIKE '%10%'; 


执行 上 面 的 语句 ， 就 可 以 查询 出 小 说 名 字 中 含有 “10” 的 信息 了 。 执 行 效果 如 图 6.13 
所 示 。 


Ricrosoft SQL Server Wanagenent Studio -=Iolx| 
文件 到 ) 编辑 下 ) 视图 CD) 查询 @) 项目 EF) 调试 @) 工具 CD) 窗口 WD 社区 人 C) 
帮助 0D 
辽 新建 查询 思 | 六 | 六 渤 访 | 刘 | 访 回归 | 癌 

时 | chapters | ?执行 DO av 弹 本 [加 3 看 | 办 车 
SQLQueryl- ))*| 

日 USE cha 
| SELECT 


口 结果 | 性 消 息 | 
[id name uthor | novelype | role | CTR | remark, 

[| 101 条 花王 狗 “史密斯 。 纸 版 。 ”NULL ”200 ”中 国 对 外 翻译 出 版 公司 
2 | 5 10S 公 富 。 。 诸 融 宁 陪 。 纸 版 。 ”NULL ”300 中国 华 恬 出 版 杜 


《| TDY-9527NISSQLSERYER2008 (... | WIDW-9527\Adninistrato. .. | chapter6 |00:00:00 | 2 行 
就 绪 行 5 列 1 Chl 


图 6.13 LIKE 关键 字 在 查询 中 的 使 用 


这 里 ,，“%10%” 就 代表 了 查询 小 说 名 称 中 是 否 含有 “10”。 如 果 想 查询 不 含有 “10” 
的 小 说 名 称 ， 那 么 ， 可 以 使 用 “NOT LIKE” 来 查询 。 读 者 可 以 自己 试 试 哦 。 


6.2.10 查询 某 一 范围 用 IN 

在 前 面 运算 符 的 学 习 里 已 经 学 习 过 IN 的 使 用 ， 它 主要 用 于 判断 某 个 值 是 否 在 指定 
的 范围 内 。 那 么 ， 什 么 时 候 使 用 IN 呢 ? 比如 : 需要 查询 年 龄 是 否 在 指定 年 龄 中 。IN 关键 
字 在 WHERE 语句 中 使 用 的 方法 如 下 : 

SELECT column namel, column name2, column name3.…- 

FROM table name 

WHERE column name IN (valuel,value2,...) 

这 里 ， 在 IN 关键 字 前 面 的 是 数据 表 中 的 列 名 ，IN 后 面 括号 中 是 具体 的 值 。 但 是 ， 要 
注意 IN 后 面 的 内 容 数 据 类 型 要 一 致 。 具 体 的 使 用 方法 请 看 示例 14。 
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【示例 14】 查询 小 说 信息 表 ， 显 示 出 作者 是 史密斯 或 者 是 欧阳 小 梅 的 小 说 信息 。 
根据 题目 要 求 ，WHERE 语句 的 写法 应 该 是 author IN( 史 密斯 "欧阳 小 梅 )， 具 体 的 语 
名 如 下 : 


USE chapter6; 
SELECT * FROM novelinfo 


WHERE author IN(' 史 密斯 '，,' 欧 阳 小 梅 ') ; 


执行 上 面 的 语句 ， 就 可 以 将 符合 条 件 的 作者 所 对 应 的 小 说 信息 查询 出 来 了 。 执 行 效果 
如 图 6.14 所 示 。 


Microsoft SQL Server Wanageseat Studio =|DI xl 
文件 中 ”编辑 FE) ”视图 WW) 查询 @) 项 目 E) 调式 0) 工具 CD) 窗口 是 社区 人 ) 
帮助 0) 
巡 新 尘 坦 询 如 | 出 | 疡 也 马上 外 | 世 回忆 | 受 瑟 
a WH | chaptere -| >》 区 归 国 沾 机 | 的 名 
SqLQueryl. s... ator (54))*| ~ X 
日 USF chapter6; 二 
SELECT * 了 RON novelinfo 
WHERE auchor IN(' 史 密斯 ', ' 欧 阳 小 梅 ') : 
> 
回 结果 | 消息 | 
| name auihor | novelype [ole | CTR | enak 
[| 0 条 花王 网 “史密斯 纸 版 。 NULL “200 “中 国 对 外 翻译 出 版 公司 
《| YTDY-952TNISSQLSERYER2008 (.. | WIDN-9527\Adninistrato. .. | chapter6 | 00:00:00 |1 行 


就 绪 行 6 列 1 Chl 4 


图 6.14 IN 关键 字 的 使 用 


从 图 6.14 的 结果 中 ， 可 以 看 出 在 小 说 信息 表 中 只 有 一 条 记录 满足 条 件 ， 即 作者 是 史 


6.2.11 根据 多 个 条 件 查 询 数据 


根据 条 件 查询 数据 ， 在 前 面 的 查询 中 也 提 到 过 ， 也 就 是 使 用 WHERE 关键 字 的 查询 语 
名 。 所 谓 按 多 个 条 件 查询 数据 ， 也 就 是 在 WHERE 后 面 放置 多 个 查询 条 件 ， 那 么 ， 这 些 条 
件 之 间 用 什么 连接 到 一 起 呢 ? 前 面 的 运算 符 可 不 是 白 学 的 啊 ， 通 常情 况 下 使 用 逻辑 运算 符 
来 连接 多 个 连接 条 件 。 

【示例 15】 查询 小 说 信息 表 ， 显 示 纸 版 并 且 点 击 率 在 100 以 上 的 小 说 信息 。 

从 题目 的 要 求 来 看 ， 需 要 在 WHERE 后 面 加 上 两 个 条 件 并 且 两 个 条 件 之 间 用 “and” 
来 连接 。 具 体 的 语句 如 下 : 

USE chapter6; 


SELECT * FROM novelinfo 
WHERE noveltype=' 纸 版 ' and CTR>100; 


执行 上 面 的 语句 ， 就 可 以 查询 出 符合 WHERE 后 面条 件 的 数据 了 。 执 行 效果 如 图 6.15 
所 示 。 
从 图 6.15 可 以 看 出 ,查询 的 结果 中 既 要 满足 小 说 类 型 是 纸 版 的 ， 又 要 满足 点 击 率 大 于 
100。 如 果 查 询 的 条 件 是 只 满足 其 中 一 条 即 可 ， 那 么 就 可 以 使 用 OR 关键 字 连 接 条 件 了 。 
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Microso: ft SQL Server Nanagenent Studio -Iolx| 
文件 于 ) ”编辑 候 视图 WW) 查询 @) 项 目 E) 调式 Wm) 工具 XY) 窗口 iD 社区 人 
帮助 中 


:及 时 喜光 如 | 让 | 也 也 加 | 包 | 克 吕 己 | 马扎 
:天 | chaptere EE 
‘SQLQueryl. s- - - at tor (54))*| ee 
日 USE chapter6; 
SELECT * FRON novelinfo 
noveltype=' 纸 版 ' and cTR>100: 


- WHERE 


4 | 101 条 花王 狗 ”史密斯 。 纸 版 NULL ”200 ”中 国 对 外 翻译 出 版 公司 
[2 | 5 108 公 富 。 ” 诸 融 字 陪 。” 纸 版 NULL ”300 ”中 国 华 桥 出 版 社 
{| WIDW-9527\MSSQLSERVER2008 (... | WIDN-9527\Adninistrato. .. | chapter6 |00:00:00 |2 行 


就 绪 行 4 列 1 Chl 六 


图 6.15 


多 条 件 的 查询 语句 


6.3 聚合 遂 数 


聚合 函数 是 数据 库 系统 中 众多 函数 中 的 一 类 ， 它 的 重要 应 用 就 是 在 查询 语句 中 使 用 。 
在 SQL Server 数据 库 中 聚合 函数 主要 包括 求 最 大 值 的 函数 MAX、 求 最 小 值 的 函数 MIN、 
求 平均 值 的 函数 AVG 以 及 求 和 函数 SUM、 求 记录 的 行 数 COUNT。 读 者 这 回 知道 了 吧 ， 
原来 聚合 函数 就 5 个。 那么 ， 这 5 个 函数 什么 时 候 用 呢 ? 就 和 它们 的 字面 意思 一 样 ， 一 般 


都 用 于 查询 和 统计 表 中 的 数据 。 
6.3.1 求 最 大 值 函数 MAX 


先 认识 一 下 求 最 大 值 的 函数 MAX， 通 常情 况 下 ， 求 最 大 值 的 列 都 要 是 数值 类 型 的 ， 
否则 就 没有 比较 的 意义 了 。 在 实际 应 用 中 ， 当 需要 得 到 商品 的 最 高 价格 、 小 说 的 最 高 点 击 


率 时 ， 都 可 以 使 用 这 个 最 大 值 函数 MAX 来 祝 你 一 辟 之 力 。 


【示例 16】 查询 小 说 信息 表 ， 并 取得 小 说 信息 表 中 最 高 的 点 击 率 信息 。 
根据 题目 要 求 ， 要 在 SELECT 语句 后 面 加 上 MAX 函数 。 具 体 语 句 如 下 : 


USE chapter6; 

SELECT MAX(CTR) FROM novelinfo; 

执行 上 面 的 语句 ， 就 可 以 查看 到 最 高 的 点 击 率 了 。 
执行 效果 如 图 6.16 所 示 。 

从 图 6.16 所 示 的 结果 可 以 看 出 , 小 说 信息 表 中 小 说 
的 最 高 点 击 率 是 300。 


6.3.2 求 最 小 值 函数 MIN 

取 最 小 值 也 是 经 常会 用 到 的 ， 比 如 : 查看 价格 最 低 
的 商品 信息 、 查 看 成 绩 最 低 的 学 生 信息 等 。 求 最 小 值 函 
数 MIN 与 MAX 的 使 用 方法 类 似 , 都 是 用 “MIN( 列 名 )” 
的 形式 来 表示 的 。 同样 , 只 能 对 数值 类 型 的 列 取 最 小 值 。 


Ricrosoft SQL Server Wessemeie [| 
文件 E) 编辑) 视图 WW 查询 @) 项目 到 ) 
调 坛 gj 工具 CD) 窗口 W) 社区 C) 帮助 只 
了 新 建 查询 四 | 让 | 孔 也 四 | 记 区 四 双 中 
时 | -hapte 


SQLQueryl- s... ator (52))* 
USE chapter6; 


SELECT NAX(CTR) FRONM nov 
4 


习 结果 | 2 消息 | 
1 300 | 


| TY-sszrwAaninistrate .. | chapteré |00:00:00 |1 行 


就 绪 行 4 列 1 Ch1l 反 4 


图 6.16 MAX 函数 的 使 用 
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下 面 就 来 看 一 下 MIN 函数 如 何 使 用 吧 。 
【示例 17】 查询 小 说 信息 表 ， 并 取得 小 说 信息 表 中 最 低 的 点 击 率 。 
根据 题目 要 求 ， 要 在 SELECT 诸 句 后 面 加 上 MIN 函数 。 具 体 语句 如 下 : 
USE chapter6; 
SELECT MIN (CTR) FROM novelinfo; 
执行 上 面 的 语句 ， 就 可 以 查看 到 点 击 率 最 低 的 小 说 信息 了 。 执 行 效果 如 图 6.17 所 示 。 
从 图 6.17 的 结果 可 以 看 出 ， 在 小 说 信息 表 中 小 说 的 最 低 点 击 率 是 100。 


6.3.3” 求 平均 值 函数 AVG 


取 平 均值 函数 就 更 有 用 了 ， 当 需要 计算 所 有 学 生 某 一 个 科目 的 平均 分 、 计 算 所 有 商品 
的 平均 价格 时 ， 就 要 考虑 使 用 求 平均 值 函数 了 。 求 平均 值 函数 用 AVG 来 表示 ， 用 来 计算 
数值 类 型 列 的 平均 值 ， 它 的 用 法 仍然 是 “AVG( 列 名 )” 下 面 就 来 看 看 示例 18 吧 。 

【示例 18】 查询 小 说 信息 表 ， 并 取得 小 说 信息 表 中 的 平均 点 击 率 。 

根据 题目 要 求 ， 要 在 SELECT 语句 后 面 加 上 AVG 函数 。 有 具体 语句 如 下 : 


USE chapter6; 
SELECT RAVG (CTR) FROM novelinfo; 


执行 上 面 的 语句 ， 就 可 以 查看 到 所 有 小 说 的 平均 点 击 率 了 。 执 行 效果 如 图 6.18 所 示 。 
从 图 6.18 所 示 的 结果 中 可 以 看 出 ， 小 说 信息 表 中 所 有 小 说 的 平均 点 击 率 是 164。 


了 microseft SQL S t .TE | 
文件 外 编辑 下) 视图 W) 查询 @) 项 目 全 ) 
调试 @) 工具 CD) 窗口 如 社区 CC) 帮助 0 
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RY WR | -hapter6 >= | 执行 &) 记 
SQLQueryl. s... ator (52))* 
UsE chapter6; 
SELECT MIN(CTR) FROM novelinfo; 
1 


文件 @) 编辑 GE) 视图) 查询 @) 项目 邓 ) 
调式 W) 工具 CD) 窗口 @ 社区 C) 姑 助 ob) 


Less 外 | 怨 辽 下 | 包 | 区 加 天 


SELECT AV' (cI) FRONM novelinfo 
«| 1 上 


口 结果 | 已 消息 | 


4 
加 结果 | 2 消息 | 


1 100 


TDpy-9s27\WAdninistrate . .| chapter6 |00:00:00 | ! 行 
就 绪 行 4 列 1 chl 才 


图 6.17 MIN 函数 的 使 用 图 6.18 AVG 函数 的 使 用 


6.3.4 求 和 函数 SUM 


SUM 是 用 来 求 列 中 数据 和 的 函数 。 它 也 是 一 个 比较 常用 的 函数 ， 比 如 ， 计 算 商品 的 价 
格 总 和 、 每 个 学 生 的 各 科 成 绩 总 和 等 。SUM 函数 也 是 对 数值 类 型 列 求 和 的 ， 它 的 用 法 是 
“SUM( 列 名 )”。 在 示例 19 中 就 演示 了 如 何 使 用 SUM 函数 求 和 。 

【示例 19】 查询 小 说 信息 表 ， 并 取得 小 说 信息 表 中 点 击 率 的 总 和 。 

根据 题目 要 求 ， 要 在 SELECT 语句 后 面 加 上 SUM 函数 。 具 体 语 句 如 下 : 


USE chapter6; 
SELECT SUM (CTR) FROM novelinfo; 


执行 上 面 的 语句 ， 就 可 以 查看 到 所 有 小 说 的 点 击 率 之 和 了 。 执 行 效果 如 图 6.19 所 示 。 
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从 图 6.19 的 查询 结果 可 以 看 出 ， 小 说 信息 表 中 小 说 的 点 击 率 之 和 是 820。 
6.3.5 求 记录 行 数 COUNT 


COUNT 函数 的 用 法 与 前 面 的 4 个 聚合 函数 略 有 不 同 ， 它 通常 是 用 来 计算 查询 结果 中 
的 行 数 。 那 么 ， 它 有 什么 用 途 呢 ? 比如 : 查询 成 绩 在 优秀 以 上 的 学 生 个 数 、 查 询 某 一 类 商 
品 的 数量 等 。COUNT 的 使 用 方法 很 简单 ， 使 用 “COUNT(*)” 就 代表 了 查询 结果 的 记录 行 
数 。 下 面 就 来 验证 一 下 COUNT(*) 的 用 法 吧 。 

【示例 20】 查询 小 说 信息 表 ， 并 取得 小 说 信息 表 中 纸 版 小 说 的 数目 。 

根据 题目 要 求 ， 要 在 SELECT 语句 后 面 加 上 COUNT 函数 ， 并 且 还 要 加 上 WHERE 条 
件 。 有 具体 语句 如 下 : 

USE chapter6; 

SELECT COUNT (*) FROM novelinfo WHERE noveltype=' 纸 版 '; 


执行 上 面 的 语句 ， 就 可 以 查看 到 纸 版 小 说 的 数目 了 。 执 行 效果 如 图 6.20 所 示 。 
从 图 6.20 的 查询 结果 可 以 看 出 ， 在 小 说 信息 表 中 有 两 本 纸 版 小 说 。 


Ls = 
icrosoft SQL Server Noneecsese iE 文件 四 编辑 下) 视图 WW 查询 @) 项 目 E) 调试 @) 工具 CD) 窗口 WD 社区 @) 


文件 @) 编辑 了 视图 WW 查询 @) 项目 中 ) dd 

调式 @) 工具 加 窗口 如 ”社区 C) 帮助 员 及 二 光 D | 记 | 翅 孙 术 | 马达 品 己 | 到 吕 

;也 新建 查 询 中 | 让 | 也 过 四 | 应 | 记 加 到 县 Ba | cheptere | 执行 DO 有 v 33 康 回 3 码 | 妈 名 
; MY HR | chaptere |》 执行 C) 三 S0Lueryl- s.. -ator (52))*| -x 


日 0sF chapr 
SELECT COUNT(*) FROM novelinfo WHERE noveltype=! 纸 版 ， 


a 


@ | YTD-9527 MSSQLSERYER2008 (.. | WIDW-9527\Adninistrato. .. | chapter6 |00:00:00 |1 行 
就 绪 


SQLQueryl. s. . -ater (52))*| 
日 USE chaprer6; 
| SELECT SUN(CTR) FROM novelinfo; 


‘ 
口 结果 | 已 消息 | 


1 sz 
[TDY-952TWAdnini strate .| chapter8 |00:00:00 |1 行 
就 绪 行 3 列 1 chl 性 


行 4 
图 6.19 SUM 函数 的 使 用 图 6.20 COUNT 函数 的 使 用 


列 1 chil Ti 


6.4 本 章 小 结 


本 章 主要 讲解 了 在 SQL 语句 中 常用 的 运算 符 、 简 单 的 查询 语句 以 及 聚合 函数 在 查询 语 
句 中 的 使 用 。 在 运算 符 部 分 主要 讲解 了 在 SQL 中 常用 的 算术 运算 符 、 比 较 运算 符 、 逻 辑 运 
算 符 、 位 运算 符 以 及 其 他 运算 符 ， 在 查询 语句 部 分 重点 讲解 了 基本 查询 语句 的 语法 形式 以 
及 儿 种 常用 的 简单 查询 方法 ;在 聚合 函数 部 分 ， 除 了 讲解 何 为 聚合 函数 ， 还 讲解 了 如 何在 
查询 语句 中 应 用 聚合 函数 来 查询 数据 。 相 信 读 者 通过 本 章 的 学 习 ， 已 经 对 查询 语句 有 所 了 
解 了 ， 那 么 ， 就 充满 信心 地 进入 下 一 章 查 询 语句 提高 的 学 习 吧 。 


6.5 本 章 习 题 


一 、 填 空 题 
1. 常用 的 运算 符 有 5 
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C. IN 
C. SUM 


C. ORDER BY 
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2. 逻辑 运算 符 包 括 
3. 查询 表 中 的 前 儿 行使 用 的 关键 字 是 
二 、 选 择 题 
1. 模糊 查询 使 用 的 关键 字 是 。 

A. AVG B. LIKE 
2. 求 和 的 聚合 函数 是 8 

A. AVG B. MIN 
3. 给 查询 结果 排序 的 关键 字 是 

A. GROUP B. TOP 
三 、 问 答题 
1. 运算 符 的 运算 优先 级 是 什么 ? 
2. 如 何 去 除 查询 结果 中 的 重复 数据 ? 


3 如何 给 查询 字段 设置 别名 ? 
四 、 操 作 题 


根据 表 6-7 所 示 的 小 说 信息 表 ， 完 成 如 下 查询 语句 。 


(1) 查询 小 说 信息 表 中 的 全 部 小 说 名 称 。 
(2) 查询 点 击 率 最 高 的 小 说 作者 名 称 。 


(3) 查询 姓 张 的 作者 所 写 的 小 说 名 称 和 小 说 类 型 。 
(4) 使 用 聚合 函数 计算 所 有 小 说 的 点 击 率 。 


:a 


D. 以 上 都 不 对 


D. COUNT 


D. 以 上 都 不 对 
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在 上 一 章 中 读者 已 经 对 查询 语句 有 所 了 解 了 ， 细 心 的 读者 可 能 会 发 现 前 面 的 查询 中 都 
针对 一 张 数据 表 。 如 果 查 询 语句 每 次 只 能 查询 一 张 数据 表 ， 那 么 ， 在 数据 库 中 的 数据 表 之 
间 就 不 会 有 任何 关系 了 。 但 实际 上 ， 这 些 数据 表 之 间 是 有 联系 的 ， 比 如 : 表 之 间 的 主 外 键 
关系 。 在 本 章 中 就 将 告诉 读者 如 何 同 时 从 多 个 数据 表 中 查询 数据 以 及 如 何 处 理 查询 的 结 
果 集 。 

本 章 的 主要 知识 点 如 下 : 

口 如 何 使 用 子 查询 

口 如 何 使 用 分 组 查询 

口 如 何 使 用 多 表 查 询 

口 如 何 对 结果 集 进 行 运算 


7.1 子 查询 


所 谓 子 查 询 ， 就 是 在 一 个 查询 语句 中 嵌 套 另 一 个 查询 。 也 就 是 说 ， 在 一 个 查询 语句 中 
可 以 使 用 另 一 个 查询 语句 中 得 到 的 查询 结果 。 这 就 好 像 是 在 网 上 报名 参加 考试 ， 首 先 要 得 
到 准 考证 的 编号 ， 然 后 用 这 个 编号 去 参加 考试 一 样 。 本 节 将 讲解 在 子 查 询 中 经 常 使 用 的 4 
个 关键 字 : IN、ANY、SOME 以 及 EXISTS。 


7.1.1 使 用 IN 的 子 查询 


IN 关键 字 在 前 面 的 运算 符 中 已 经 说 过 了 , 就 是 用 来 判断 某 个 列 是 否 在 某 个 范围 内 。 在 
子 查询 中 ， 通 常用 在 查询 结果 的 前 面 ， 用 于 判断 查询 结果 中 是 否 有 符合 条 件 的 数据 。 具 体 
的 用 法 如 下 : 

SELECT column namel, column name2,... 

FROM table namel 


WHERE column name IN(SELECT column namell FROM table name2 WHERE 
conditions); 


其 中 : IN 关键 字 后 面 的 查询 就 是 一 个 子 查询 ， 并 且 在 IN 后 面 的 查询 语句 只 能 返回 一 
列 值 。 另 外 ，IN 后 面 查询 语句 返回 值 的 数据 类 型 要 与 IN 前 面 列 的 数据 类 型 相 兼容 才 可 以 
哟 。 此 外 ， 在 WHERE 语句 中 不 仅 可 以 有 子 查询 ， 也 可 以 放置 一 些 其 他 的 查询 条 件 ， 这 些 
查询 条 件 之 间 用 逻辑 运算 符 相关 联 就 可 以 了 。 

在 开始 学 习 IN 关键 字 的 子 查询 的 示例 之 前 ， 先 来 创建 一 下 本 章 要 使 用 的 数据 库 和 数 
据 表 。 本 章 要 使 用 的 数据 库 命名 为 chapter7。 在 数据 库 chapter7 中 ， 创 建 晚 会 节目 信息 表 
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以 及 节目 类 型 信息 表 。 晚 会 节目 信息 表 中 应 该 有 哪些 字段 呢 ? 想 想 每 年 我 们 都 要 看 的 春节 
晚会 ， 就 应 该 想到 一 些 字 段 内 容 了 吧 。 应 该 有 节目 名 称 、 节 目 类 型 、 演 员 、 是 否 伴舞 以 及 
节目 时 间 等 字段 。 节 目 类 型 表 中 记录 节目 的 类 型 ， 也 便于 后 面 多 表 查 询 时 使 用 。 节 目 类 型 
表 中 的 字段 包括 类 型 编号 和 类 型 名 称 。 有 具体 的 表 结 构 如 表 7-1 和 表 7-2 所 示 。 


表 7-1 节目 信息 表 (programinfo) 


编号 列 名 数据 类 型 中 文 释义 
| id INT 编号 
2 name VARCHAR(20) 节目 名 称 
3 actor VARCHAR(20) 演员 
4 programtype INT 节目 类 型 编号 
5 programtime DECIMAL (4,2) 节目 时 长 
6 author VARCHAR(20) 节目 作者 
多 remark VARCHAR(200) 备注 
表 7-2 节目 类 型 信息 表 (typeinfo) 
编号 列 名 数据 类 型 中 文 释义 
1 id INT 编号 
2 name VARCHAR(20) 节目 名 称 


在 数据 库 chapter7 中 ,创建 节目 信息 表 和 节目 类 型 信息 表 的 语句 如 下 : 


USE chapter7; 
CREATE TABLE programinfo 
( 
id INT identity(1,1) PRIMARY KEY, 
name VARCHAR(20), 
actor VARCHAR(20), 
programtype INT, 
programtime DECIMAL(4,2), 
author VARCHAR(20), 
remark VARCHAR(200) 


; 
CREATE TABLE typeinfo 
( 
id INT identity(1,1) PRIMARY KEY, 
name VARCHAR(20) 
地 


执行 上 面 的 语句 , 就 可 以 创建 节目 信息 表 和 节目 类 型 信息 表 了 。 创建 好 这 两 张 数 据 表 ， 
分 别 向 这 两 张 表 中 输入 表 7-3 和 表 7-4 所 示 的 数据 。 


表 7-3 节目 信息 表 中 的 数据 


团结 就 是 力量 
三 和 三 让 
江南 style 


st 
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名 称 节目 时 长 节目 作者 备注 
变换 扑克 3 3 10 王菲 菲 无 
踩 球 5 15 章 欢 无 
我 们 是 一 家 人 陈 新 无 


表 7-4 ”节目 类 型 信息 表 中 的 数据 


向 这 两 张 数 据 表 中 添加 数据 的 语句 如 下 : 


USE chapter7; 

INSERT INTO programinfo VALUES (' 我 爱 我 家 ', ' 刘 品 ' ,1,2.7, ' 周 名 ', ' 无 ')， 
(' 团 结 就 是 力量 ',' 杨 晓 等 ' ,1, 4, ' 贾 羽 ',' 合 唱 ')， 
(0 三 大 三 夜 W7 二 辽 冯 rH5r 际 默 同 天 史 
(Eye "未 有 27 
sa Fi Sl 
(' 踩 球 ',' 章 欢 ', 5,15, ' 章 欢 ', ' 无 ' )， 
(' 我 们 是 一 家 人 ',' 陈 新 等 ',2,10, ' 陈 新 ',' 无 ')， 

INSERT INTO typeinfo VALUES (' 歌 舞 类 ')，,(' 语 言 类 '),(' 魔 术 类 ')，,(' 访 谈 类 ')，,(' 杂 技 

类 ')，; 

执行 上 面 的 语句 ， 这 两 张 数据 表 中 的 数据 就 创建 好 了 。 

有 了 这 两 张 数 据 表 ， 就 可 以 使 用 本 章 的 各 种 查询 语句 尽情 地 查询 了 。 下 面 就 来 看 IN 
关键 字 如 何在 子 查询 中 使 用 吧 。 

【示例 1】 使 用 子 查询 来 查询 歌舞 类 和 语言 类 节目 。 

根据 题目 要 求 ， 在 WHERE 语句 的 子 查询 中 查询 类 型 信息 表 中 的 类 型 名 称 。 具 体 语句 
如 下 : 

USE chapter7; 

SELECT * FROM programinfo 

WHERE programtype IN (SELECT id FROM typeinfo WHERE name=' 歌 舞 类 ' OR name= 

' 语 言 类 ' ) ; 

执行 上 面 的 语句 ， 就 可 以 将 节目 信息 表 中 所 有 歌舞 类 和 语言 类 的 节目 全 部 查询 出 来 。 
执行 效果 如 图 7.1 所 示 。 

从 图 7.1 所 示 的 查询 结果 中 可 以 看 出 ， 把 节目 类 型 编号 是 1 或 2 的 信息 全 部 都 查询 出 
来 了 (1 代表 歌舞 类 ，2 代表 语言 类 )。 也 就 是 说 ， 子 查询 的 结果 就 应 该 是 1 和 2。 如 果 想 
查询 不 是 歌舞 类 和 语言 类 的 节目 时 ， 可 以 使 用 “NOT IN” 关 键 字 来 查询 。 读 者 这 回 体会 到 
子 查 询 的 便利 了 吧 。 如 果 没 有 子 查询 ， 那 么 ， 这 个 查询 用 第 6 章 的 知识 是 无 法 完成 的 。 


7.1.2 使 用 ANY 的 子 查询 
ANY 关键 字 也 是 在 子 查询 中 经 常 使 用 的 ， 通 常 都 会 使 用 比较 运算 符 来 连接 ANY 得 到 


wa 
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Aicrosoft SQL Server Nanagenent Studio =Iolx| 

文件 外 编辑 信 ) 视图 C) 查询 和 @) 项 目 E) 调式 @) 工具 CD) 窗口 @) 社区 C) 帮助 00 

也 新 建 查询 W | 出马 也 怠 包 | 区 回忆 | 怠 操 

于 WR | chepter7 了 1? 执行 四 》  v 弹 时 国 | 福 攀 | 办 圈 | 的 | 三 全 | 幸 奉 | 急电 
ater (53))*| WIDW-9527\MN... prograninfo | TIY-952TWI . .dbo. typeinfo | = 
7 


ON programinfo 
上 wHERE progremtype IN (SELECT id FRONM typeinfo WHERE name-' 歌 舞 类 ，oR name-' 语 言 类 '); 


团结 就 是 力量 杨 晓 等 
三 天 三 夜 王 晓 可 
江南 style 朱 征 等 
我 们 是 一 家 人 陈 新 等 


| WIDN-9527\MISSQLSERVER2008 (... | WIDN-9527\Adninistrato., .| chapter7 | 00:00:00 |5 行 
行 6 列 1 Ch1l 


图 7.1 IN 在 子 查询 中 的 使 用 


的 结果 。 它 可 以 用 于 比较 某 一 列 的 值 是 否 全 部 都 大 于 ANY 后 面子 查询 中 查询 的 结果 ， 或 
者 小 于 ANY 后 面子 查询 中 的 查询 的 结果 等 操作 。 使 用 ANY 的 语法 如 下 : 

SELECT column namel, column name2,... 

FROM table namel 

WHERE column name operator ANY(SELECT column namel]l FROM table name2 WHERE 

conditions); 

这 里 ，operator 就 是 用 于 列 与 ANY 后 面 所 有 的 查询 结果 进行 比较 的 运算 符 。 运算 符 包 
i “<=” 等 。 

【示例 2】 使 用 子 查 询 来 查询 语言 类 节目 时 长 大 于 歌舞 类 节目 时 长 的 节目 信息 。 

根据 题目 要 求 ， 要 首先 查询 出 所 有 歌舞 类 的 节目 时 长 。 具 体 语句 如 下 : 

USE chapter7; 


SELECT programtime FROM programinfo 
WHERE programtype = (SELECT id FROM typeinfo WHERE name=' 歌 舞 类 ' ); 


执行 上 面 的 语句 ， 就 可 以 将 节目 信息 表 中 所 有 歌舞 类 节目 的 时 长 查询 出 来 。 下 面 就 要 
开始 使 用 语言 类 的 节目 时 长 与 其 比较 了 。 这 里 ， 为 了 让 读者 能 够 更 好 地 理解 ANY 关键 字 
的 使 用 ， 直 接 在 查询 中 使 用 语言 类 节目 的 编号 “2” 进 行 查询 。 查 询 语句 如 下 : 

USE chapter7; 

SELECT * FROM programinfo 

WHERE programtime >ANY 

(SELECT programtime FROM programinfo 

WHERE programtype = (SELECT id FROM typeinfo WHERE name=' 歌 舞 类 ' ) 

AND programtype=2; 

执行 上 面 的 语句 ， 就 可 以 查询 语言 类 节目 时 长 大 于 歌舞 类 节目 时 长 的 信息 了 。 执 行 效 
果 如 图 7.2 所 示 。 

从 图 7.2 所 示 的 查询 结果 中 可 以 看 出 ，ANY 前 面 的 运算 符 就 “>” 就 代表 了 对 ANY 后 
面子 查询 的 结果 中 任意 值 进行 是 否 大 于 的 判断 。 如 果 要 判断 小 于 可 以 使 用 “<” 判断 不 等 
于 可 以 使 用 “! =”。 与 ANY 关键 字 功 能 一 样 的 是 ALL 关键 字 ， 在 实际 应 用 中 使 用 哪个 关 


.154 。 


mi croso: ft SQL Server Wanagenent Studie =Io| x| 
文件 于 ”编辑 下 ) 视图 WV) 查询 @) 项 目 @) 调式 Wm) 工具 XY) 窗口 t) 社区 CC) 帮助 00 
卫 新建 查询 思 | 让 | 轴 也 怠 | 记 | 蕊 中 己 | 又 后 
: BY a | chapter7 -| 执行 QW) 》 mv 中 至 国 | 3 本 | 给 加 | 瓜 
‘SQLQueryl.s... a ter (53))*| 人 TDY-952TWL . .prograninfo | x 
日 05E chapter7; 
SELECT * FRON programinfo 
WHERE programt ime >ANY 
(SELECT programtime 了 RON programinfo 
WHERE programcype = (SELECT id FROM cypeinfo WHERE name=' 歌 兽 类 ')) 
FAND programtype=2; 


器 结果 | 2 消息 | 


这 ] name actor | progamgpe | programtime | author | remark 
1 江南 style 朱 两 等 2 700 刘 青 无 
7 我 们 是 一 家 人 陈 新 等 2 1000 阵 新 无 


回 查询 … | YTDY-952TWISSQLSERVER2006 (〔 .| WIDW-9527\Adninistrato.. .| chapter7 | 00:00:00 |2 行 
匹配 : ( 行 8 列 ! ch1 wwe 


图 7.2 ANY 在 子 查询 中 的 使 用 


键 字 都 可 以 。 读 者 可 以 在 上 面 的 示例 中 尝试 应 用 一 下 ALL 关键 字 ， 看 看 查询 结果 是 否 一 
样 呢 ? 


7.1.3 使 用 SOME 的 子 查询 


SOME 关键 字 的 用 法 与 ANY 的 用 法 非常 类 似 ， 但 是 意义 却 有 所 不 同 。SOME 通常 用 
于 比较 满足 查询 结果 中 的 任意 一 个 值 ， 而 ANY 是 要 满足 所 有 值 才 可 以 。 因 此 ， 在 实际 应 
用 中 ， 可 要 特别 注意 查询 条 件 哟 。SOME 在 子 查 询 中 应 用 的 语句 如 下 : 

SELECT column namel, column name2,... 

FROM table namel 

WHERE column name operator SOME (SELECT column namell FROM table name2 

WHERE conditions); 

这 里 ，operator 就 是 用 于 列 与 SOME 后 面 任意 一 个 查询 结果 值 进行 比较 的 运算 符 。 运 
算 符 包括 “= “> “< “1 =” “>=” 和 “<=” 等 。 

【示例 3】 查询 节目 信息 表 ， 并 使 用 SOME 关键 字 选 出 所 有 歌舞 类 和 杂技 类 的 节目 信 

根据 题目 要 求 ， 从 节目 类 型 表 中 查询 出 歌舞 类 和 杂技 类 节目 的 类 型 编号 ， 然 后 再 使 用 

节目 信息 表 中 的 类 型 编号 与 其 比较 。 查 询 语句 如 下 : 

USE chapter7; 

SELECT * FROM programinfo 


WHERE programtype=SOME (SELECT id FROM typeinfo WHERE name=' 歌舞 类 ' OR name= 
"杂技 类 ' ) ; 


执行 上 面 的 语句 ， 就 可 以 查询 节目 信息 表 中 的 歌舞 类 和 杂技 类 的 节目 信息 。 执 行 效果 
如 图 7.3 所 示 。 

从 图 7.3 所 示 的 查询 结果 中 可 以 看 出 ， 所 有 的 歌舞 类 和 杂技 类 节目 全 部 查询 出 来 了 。 
读者 是 否 想到 了 前 面 有 一 个 关键 字 也 可 以 完成 与 其 相同 的 功能 。 没 错 ， 这 就 是 IN 关键 字 。 
也 就 是 说 ， 当 在 SOME 运算 符 前 面 使 用 “=” 时 ， 就 代表 了 IN 关键 字 的 用 途 。 


Ep 
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Si =5Ix 
文件 时) 编辑 于 ) 视图 查询 @) 项 目 E) 调式 @) 工具 Q) 窗口 WD 社区 人 C) 帮助 0D 
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-ator (53))*| WIDW-9527WN.. prograninfo | WIDW-9527\N... dbo. typeinfo | =x 


nfo 
SELECT id FROM typeinfo WHERE name=, 歌 舞 类 ，cr 


团结 就 是 力量 杨 晓 等 ”1 
三 天 三 夜 王 晓 可 1 
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图 7.3 SOME 在 子 查询 中 的 使 用 


7.1.4 使 用 EXISTS 的 子 查询 


EXISTS 关键 字 代 表 “ 存 在 ”的 意思 。 它 应 用 于 子 查 询 中 ， 只 要 子 查 询 返 回 的 结果 为 
空 ， 那 么 ， 返 回 就 是 true， 否 则 就 是 false。 通 常情 况 下 EXISTS 关键 字 用 在 WHERE 语句 
中 。EXISTS 在 子 查询 中 使 用 的 语法 如 下 

SELECT column namel, column name2,... 

FROM table namel 

WHERE EXISTS (SELECT column namell FROM table name2 WHERE conditions); 

这 里 ， 当 EXISTS 后 面 的 查询 语句 能 够 查询 出 数据 ， 那 么 ， 就 查询 出 所 有 符合 条 件 的 
数据 ， 和 否则， 就 不 输出 任何 数据 。 

【示例 4】 查询 节目 信息 表 ， 并 选 出 与 节目 类 型 表 中 一 致 的 节目 信息 。 

为 了 能 够 体现 查询 效果 ， 在 节目 信息 表 中 添加 数据 如 表 7-5 所 示 。 


表 7-5 添加 节目 信息 表 中 的 数据 


| 名称 | 演员 | 类型 | 节目 时 长 | 节目 作者 | 


nm | mw sw | wo | ax | 


向 表 中 添加 这 条 数据 后 ， 就 可 以 开始 写 语句 了 。 查 询 语 句 如 下 : 

USE chapter7; 

SELECT * FROM programinfo 

WHERE EXISTS (SELECT * FROM typeinfo WHERE programinfo. 

programtype=typeinfo.id); 

执行 上 面 的 语句 ， 就 可 以 查询 出 所 有 符合 条 件 的 数据 了 。 执 行 效果 如 图 7.4 所 示 。 

从 图 7.4 所 示 的 查询 结果 中 可 以 看 出 ， 并 没有 将 后 来 添加 的 记录 查询 出 来 ， 主 要 是 因 
为 该 记录 的 节目 类 型 编号 在 节目 类 型 信息 表 中 不 存在 。 当 然 ， 针 对 同一 个 题目 ， 查 询 语句 
也 可 以 有 多 种 写法 ， 这 里 ， 只 是 演示 EXISTS 关键 字 的 用 法 而 已 。 读 者 也 可 以 使 用 其 他 的 
programtype 是 什么 意思 呢 ? 为 什么 不 直接 写 programtype 呢 ? 那 是 因为 在 这 个 查询 语句 中 
用 了 两 张 数据 表 ， 为 了 能 够 区 分 每 个 列 的 来 源 ， 要 在 列 前 面 加 上 该 列 所 在 的 表 名 。 这 种 用 
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图 7.4 EXISTS 在 子 查询 中 的 使 用 
法 还 将 在 本 章 后 面 的 小 节 中 用 到 ， 请 读者 认真 体会 。 


7.2 分 组 查询 


在 学 习 分 组 查询 之 前 ， 读 者 先 要 弄 清楚 什么 是 分 组 。 在 现实 生活 中 , 经 常会 用 到 分 组 ， 
比如 ; 扫 雪 时 经 常会 将 一 个 班级 分 成 儿 个 小 组 ， 分 别 完 成 不 同 的 扫 雪 任务 ， 开 运动 会 时 ， 
经 常会 将 每 类 比赛 报名 的 运动 员 进行 分 组 ， 分 别 进行 小 组 中 的 预赛 。 在 数据 库 中 的 分 组 也 
是 同一 个 意思 ， 将 数据 按照 一 定 条 件 进行 分 组 ， 然 后 统计 每 组 中 的 数据 。 


7.2.1 分 组 查询 介绍 


分 组 查询 主要 应 用 在 数据 库 的 统计 计算 中 。 在 任何 一 个 应 用 软件 中 都 不 会 缺少 统计 查询 
的 ， 因 此 , 分 组 查询 是 至 关 重 要 的 。 分 组 查询 使 用 GROUP BY 子 句 来 完成 , 具体 的 语法 如 下 : 

SELECT column namel, column name2,... 

FROM table namel 

[WHERE] conditions 

GROUP BY column namel, column name2.… 

[HAVING] conditions 

[ORDER BY] column namel, column name2....; 


其 中 : 

口 GROUP BY: 分 组 查询 的 关键 字 。 在 其 后 面 写 的 是 按 其 分 组 的 列 名 ， 并 且 可 以 按 
照 多 列 进行 分 组 。 

口 HAVING: 在 分 组 查询 中 使 用 条 件 的 关键 字 。 该 关键 字 只 能 用 在 GROUP BY 语句 
后 面 。 它 的 作用 与 WHERE 语句 类 似 ， 都 表示 查询 的 条 件 。 但 是 ， 在 执行 效率 上 
略 有 不 同 。 在 7.2.3 小 节 中 将 详细 讲解 HAVING 的 用 法 。 

在 上 面 的 语法 中 ，WHERE、HAVING 和 ORDER BY 都 是 可 以 省 略 的 ， 根 据 实际 需要 

自行 添加 就 可 以 了 。 另 外 ， 在 分 组 查询 还 经 常会 使 用 到 聚合 函数 。 并 且 ， 在 分 组 查询 中 
SELECT 语句 后 面 还 有 一 些 说 道 呢 ! 那 就 是 ， 在 SELECT 语句 后 面具 能 出 现 聚 合 函数 和 
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GROUP BY 语句 后 面 的 列 名 。 这 一 点 非常 重要 ， 千 万 要 记 住 啊 ! 
7.2.2 ”聚合 函数 在 分 组 查询 的 应 用 


读者 应 该 还 记得 聚合 函数 都 包括 什么 吧 ? 没 错 ， 就 是 MAX、MIN、COUNT、AVG 和 
SUM 这 5 个 。 它 们 在 分 组 查询 中 有 什么 用 呢 ? 想 一 想 ， 其 实用 处 还 是 挺 多 的 ， 比 如 : 查看 
一 下 谁 是 小 组 第 一 ， 在 每 一 类 小 说 中 哪 本 小 说 销量 最 差 等 信息 。 既 然 这 么 有 用 ， 就 让 我 们 
一 起 尝试 一 下 吧 。 

【示例 S】 查询 节目 信息 表 中 ， 每 类 节目 中 时 间 的 最 短 值 。 

根据 题目 要 求 , 可 以 按照 节目 信息 表 中 的 节目 类 型 进行 分 组 , 然后 使 用 MIN 函数 来 获 
取 时 间 最 短 的 节目 。 语 句 如 下 : 

USE chapter7; 

SELECT programtype,MIN (programtime) 

FROM programinfo 

GROUP BY programtype; 

执行 上 面 的 语句 ， 就 可 以 按照 指定 的 分 组 找到 节目 时 间 的 最 短 值 。 执 行 效果 如 图 7.5 
所 示 。 

从 图 7.5 所 示 的 查询 结果 中 可 以 看 出 ， 在 节目 ET 
信息 表 中 共有 5 种 类 型 的 节目 ， 每 种 类 型 节目 的 最 。 wease 名 
短 值 显示 在 每 种 类 型 的 后 面 。 那 么 ， 如 果 要 查看 语 。 入 半 vs 
言 类 节目 中 时 间 最 短 的 是 哪 一 个 节目 ， 应 该 怎么 查 
询 呢 ? 可 以 使 用 在 上 一 节 中 介绍 的 子 查询 来 完成 。 
下 面 就 让 我 们 见识 一 下 示例 6 的 查询 方法 吧 。 

【示例 6】 查看 语言 类 节目 中 时 间 最 短 的 节目 
信息 。 

在 写 具 体 的 查询 语句 之 前 ， 先 要 查看 一 下 语言 
类 最 短 的 节目 时 间 ， 然 后 再 查看 这 个 最 短 时 间 对 应 。 eee le ||| J 
节目 信息 。 为 查询 语句 简单 ， 假 设 已 经 知 i 
np mp 知道 操 7 5 聚合 函数 MIN 在 子 查 询 中 的 使 用 

USE chapter7; 

SELECT * FROM programinfo 

WHERE programtime=(SELECT MIN (programtime) FROM programinfo WHERE 

programtype=2); 

执行 上 面 的 语句 ， 就 可 以 查看 题目 要 求 的 节目 信息 了 。 执 行 效果 如 图 7.6 所 示 。 

从 图 7.6 所 示 的 查询 结果 中 可 以 看 出 , 查询 出 了 语言 类 节目 时 间 长 度 是 7 的 节目 信息 。 
同样 ， 也 可 以 使 用 MAX 函数 查看 出 节目 时 间 最 长 的 节目 信息 。 


7.2.3 在 分 组 查询 中 也 可 以 使 用 条 件 


在 讲解 分 组 查询 的 语法 时 ， 读 者 已 经 知道 了 在 分 组 查询 中 也 是 可 以 加 上 条 件 的 。 在 分 
组 查询 中 使 用 条 件 ， 既 可 以 使 用 WHERE 子 句 也 可 以 使 用 HAVING 子 句 。 它们 究 竞 有 什么 
区 别 呢 ? 它们 的 区 别 很 简单 ， 那 就 是 根据 它们 在 查询 语句 的 位 置 决定 的 。WHERE 子 句 要 
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图 7.6 聚合 函数 在 子 查 询 中 的 使 用 


放 在 GROUP BY 子 句 之 前 ， 也 就 是 说 它 能 够 先 按 条 件 筛选 数据 后 ， 再 对 数据 进行 分 组 ; 
HAVING 子 句 要 放 在 GROUP BY 子 句 之 后 ， 也 就 是 说 要 对 数据 先进 行 分 组 ， 然 后 再 对 其 
按 条 件 进 行 数据 筛选 。 读 者 既然 知道 了 它们 在 执行 查询 时 的 区 别 ， 那 么 ， 哪 个 子 句 效率 更 
高 一 些 呢 ? 当然 是 WHERE 子 句 了。 因此 ， 在 分 组 查询 的 实际 应 用 中 ， 使 用 WHERE 语句 
作为 条 件 判 断 是 用 得 比较 多 的 。 另 外 ， 还 有 一 点 最 重要 的 区 别 就 是 ， 使 用 HAVING 语句 作 
为 条 件 ， 条 件 后 面 的 列 只 能 是 在 GROUP BY 子 句 后 面 出 现 过 的 列 。 

【示例 7】 分别 使 用 WHERE 和 HAVING 子 句 进行 分 组 查询 。 按 节目 类 型 进行 分 组 并 
查询 出 姓 王 的 演员 所 表演 的 节目 类 型 编号 以 及 节目 的 数量 。 

在 本 题 中 ， 分 组 的 列 是 节目 类 型 编号 ， 条 件 是 姓 王 的 演员 。 使 用 WHERE 和 HAVING 
子 句 查询 如 下 。 

(1) 使 用 WHERE 语句 作为 条 件 判断 


USE chapter7; 

SELECT programtype,count (*) FROM Programinfo 

WHERE actor like ' 王 %' 

GROUP BY programtype 

执行 上 面 的 语句 ， 就 可 以 查看 到 姓 王 的 所 表演 的 节目 类 型 编号 以 及 节目 个 数 。 执 行 效 
果 如 图 7.7 所 示 。 

从 图 7.7 所 示 的 查询 结果 中 可 以 看 出 ， 含 有 姓 王 的 演员 的 节目 类 型 编号 是 1 和 3， 并 
且 每 类 节目 都 具有 1 个 。 

(2) 使 用 HAVING 子 句 作为 条 件 判 断 


USE chapter7; 

SELECT programtype,count (*) FROM programinfo 
GROUP BY programtype,actor 

HAVING actor like ' 王 %' 


执行 上 面 的 语句 ， 也 可 以 完成 与 (1) 相同 的 效果 。 执 行 效果 如 图 7.8 所 示 。 

从 图 7.8 所 示 的 查询 结果 中 可 以 看 出 ， 确 实 是 与 图 7.7 的 查询 结果 一 致 的 。 但 是 ,在 
GOURP BY 子 句 后 面 出 现 了 两 个 列 ， 那 是 因为 如 果 不 出 现 两 个 列 就 无 法 在 HAVING 子 句 
中 对 actor 列 进行 判断 了 。 因 此 ， 在 这 种 情况 下 ， 一 般 会 选择 使 用 WHERE 子 句 来 完成 。 
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7.2.4 分 组 查询 结果 也 能 排序 


在 分 组 查询 语法 中 最 后 一 个 子 句 ORDER BY 就 是 对 查询 结果 进行 排序 的 .ORDER BY 
子 句 会 放 到 所 有 查询 子 句 的 最 后 出 现 ， 表 示 对 查询 结果 进行 排序 。 在 排序 的 时 候 也 是 按照 

列 的 升序 和 降序 排列 的 , 并 且 , 也 可 以 同时 对 多 个 列 进行 排序 。 但 是 在 分 组 查询 中 , ORDER 
BY 后 面 的 列 也 要 是 GROUP BY 子 句 中 出 现 过 的 列 才 可 以 。 

【示例 8】 使 用 ORDER BY 子 句 进行 查询 。 查 询 出 每 种 类 型 节目 所 占用 的 时 间 ， 并 按 
节目 类 型 编号 进行 降序 排列 。 

查询 所 占用 的 时 间 可 以 使 用 聚合 函数 SUM 来 完成 。 具 体 的 查询 语句 如 下 : 

USE chapter7; 


SELECT programtype,SUM(programtime) FROM Programinfo 
GROUP BY programtype 
ORDER BY programtype DESC; 


执行 上 面 的 语句 ， 就 可 以 将 每 种 类 型 节目 所 占用 的 时 间 查 询 出 来 ， 并 对 其 节目 类 型 编 
进行 降序 排列 。 执 行 效果 如 图 7.9 所 示 。 
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图 7.9 ORDER BY 子 句 在 分 组 查询 中 的 使 用 
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从 图 7.9 所 示 的 查询 结果 中 可 以 看 出 ， 查 询 结 果 按照 节目 类 型 列 (programtype》 进 行 
了 降序 排列 。 


7.3 多 表 查 询 


在 前 面 的 查询 中 ， 已 经 涉及 到 两 张 表 之 间 的 查询 了 。 实 际 上 ， 也 可 以 实现 涉及 更 多 张 
表 的 查询 ， 只 要 表 与 表 之 间 有 一 些 相 关 的 内 容 ， 就 可 以 放 在 一 起 查询 。 多 表 查 询 主 要 内 容 
包括 表 之 间 的 自 连接 查询 、 外 连接 查询 以 及 内 连接 查询 。 在 实际 应 用 中 ， 儿 乎 每 一 个 软件 
系统 中 都 是 会 用 到 多 表 查 询 的 。 


7.3.1 第 卡 尔 积 


笛 卡尔 积 是 针对 一 种 多 表 查 询 的 特殊 结果 来 说 的 ， 它 的 特殊 之 处 就 在 于 多 表 查 询 时 没 
有 指定 查询 条 件 ， 查 询 的 是 多 个 表 中 的 全 部 记录 。 那 么 ， 读 者 可 以 想 一 想 ， 如 果 不 指定 查 
询 条 件 ， 结 果 会 是 什么 样 的 呢 ? 是 全 部 数据 的 罗列 ， 还 是 把 全 部 数据 都 挤 到 一 行 中 ， 或 者 
是 其 他 的 形式 ? 不 管 怎 么 样 ， 任 何事 情 还 是 要 以 事实 说 话 的 ， 下 面 就 用 示例 9 来 看 看 笛 卡 
尔 积 究竟 是 什么 样 的 。 

【示例 9】 不 使 用 任何 条 件 查 询 节 目 信息 表 和 节目 类 型 信息 表 中 的 全 部 数据 。 

从 题目 来 看 ， 该 查询 语句 只 需要 在 SELECT 语句 中 用 “* ”代替 所 有 列 ， 并 在 FROM 
后 面 列 出 两 张 表 的 名 字 。 查 询 语句 如 下 : 


USE chaper7; 
SELECT * FROM programinfo,typeinfo; 


执行 上 面 的 语句 ， 就 可 以 见证 笛 卡 尔 积 究 竟 是 什么 样 的 了 。 执 行 效果 如 图 7.10 所 示 。 

从 图 7.10 所 示 的 查询 结果 可 以 看 出 ， 查 询 结果 中 的 数据 好 多 啊 ， 共 有 9 列 、40 行 。 
那么 ， 这 个 行 数 和 列 数 是 怎么 从 两 张 表 的 数据 中 得 到 的 呢 ? 请 读者 看 一 看 ， 列 数 是 否 是 两 
张 数 据 表 中 列 的 总 和 ， 行 数 是 否 是 两 张 数据 表 中 行 的 乘积 呢 ? 在 节目 信息 表 中 ， 共 有 7 列 
8 行 ， 在 节目 类 型 信息 表 中 ， 共 有 2 列 5 行 。 那 么 ， 读 者 可 以 将 其 列 和 行 按照 要 求 进行 加 
和 乘 的 运算 ， 看 看 是 不 是 得 到 了 9 和 40 呢 。 没 错 ， 笛 卡尔 积 的 结果 就 是 每 张 表 中 列 的 和 、 
行 的 乘积 。 


全 注意 : 在 使 用 多 表 连 接 查询 时 ， 一 定 要 设 定 查询 条 件 ， 否 则 就 会 产生 笛 卡 尔 积 。 笛 卡尔 
积 会 降低 数据 库 的 访问 效率 。 因此， 每 一 个 数据 库 的 使 用 者 都 要 避免 查询 结果 中 
笛 卡 尔 积 的 产生 。 


7.3.2 ”同一 个 表 的 连接 一 一 自 连接 


查询 语句 不 仅 可 以 查询 多 张 表 中 的 内 容 ， 还 可 以 同时 连接 多 次 同一 张 数 据 表 。 那 么 ， 
把 这 种 同一 张 表 的 连接 称 为 自 连接 ， 也 就 是 自己 连接 自己 的 意思 。 但 是 ， 在 查询 时 要 分 别 
为 同一 张 表 设置 不 同 的 别名 。 下 面 就 通过 示例 10 来 学 习 一 下 什么 是 自 连接 。 

【示例 10】 使 用 自 连接 查询 。 查 询 出 演出 者 和 节目 作者 是 同一 个 人 的 节目 信息 。 

从 题目 的 信息 来 看 ， 演 出 者 和 节目 作者 都 在 节目 信息 表 中 ， 比 较 这 两 个 字段 相等 就 得 
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6 本 
文件 E) ”编辑 下 ) 视图 WD 查询 @) 项 目 E) 调试 四) 工具 CO) 窗口 @ 社区 CC) 帮助 0D 


县 站 | 访 | 古 馆 名 | 访 | 蕊 回 弓 | 加 
时 组 | chapter7 -| 执行) 》 和 吕 癌 国 |3” 续 | 冰 央 | 如 | 三 
“SQLQueryl. s.. r (53))#| WIDN-9527\N.,. prograninfo Xx 


日 usE chapter7; 
| SELECT * FRON programinfo,typeinfo; 


.| 9TDW-952T\Adninistrate .| chapterT | 00:00:00 
职 绪 行 6 列 1 ch1l Ins 天 


图 7.10 产生 笛 卡 尔 积 


用 同一 个 表 了 。 查 询 语 句 如 下 : 


USE chapter7; 

SELECT a.name, a.actor, a.author, a.programtime FROM programinfo a, 
programinfo b 

WHERE a.actor=b.author; 


执行 上 面 的 语句 ， 就 可 以 将 演出 者 和 作者 是 同一 个 人 的 节目 信息 查询 出 来 了 。 执 行 效 
果 如 图 7.11 所 示 。 


文件 时 编辑 于 ) 视图 WD 查询 @) 项 目 @) 调试 @) 工具 人 窗口 轨 社区 C) 帮助 0 


;也 新 建 查询 | 让 | 轴 包 可 | 马上 | 区 四 马 | 受 忆 

: By WH | chapter7 "| 执行 如 器 加 
“SQLQueryl. s... ator (53))*| WIDW-9527\N... prograninfo | WIDN-9527\N... dbo. typeinfo | 
BUSE chapter7; 


由 SELECT a.name, a.actor, a.author,a.programtime FRON programinfo a, programinfo b 
上 waERE a.actor=b.author; 


回 结果 | 5 消息 | 


| YTDY-sszTWAaninistrate . .| chapter7 |00:00:00 |3 行 
行 5 列 1 Chl 


图 7.11 自 连接 的 使 用 
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从 图 7.11 所 示 的 查询 结果 中 可 以 看 出 ， 结 果 都 是 演出 者 (actor) 和 作者 (author) 是 
同一 个 人 的 节目 信息 。 请 读者 思考 一 个 问题 ， 如 果 在 SELECT 后 面 只 写 一 个 “*”， 那 么 查 
询 结果 会 是 什么 样 呢 ? 读者 可 以 尝试 一 下 ,看 看 结果 是 否 是 显示 两 次 节目 信息 表 的 数据 呢 。 


7.3.3 ”能 查询 出 额外 数据 的 连接 一 一 外 连接 


在 前 面 的 所 有 查询 语句 中 ， 查 询 结果 全 部 都 是 需要 符合 条 件 才能 够 查询 出 来 。 换 句 话 
说 ， 如 果 执 行 查询 语句 后 没有 符合 条 件 的 结果 ， 那 么 ， 在 结果 中 就 不 会 有 任何 记录 。 在 本 
小 节 中 要 学 习 的 外 连接 ， 会 给 你 带 来 不 同 的 查询 效果 哟 。 通 过 外 连接 查询 ， 可 以 在 查询 出 
符合 条 件 的 结果 后 还 能 显示 出 某 张 表 中 不 符合 条 件 的 数据 。 外 连接 查询 包括 左 外 连接 、 碳 
外 连接 以 及 全 连接 。 先 来 具体 看 一 下 外 连接 查询 的 基本 语法 吧 。 
SELECT column namel, column name2,..... 
FROM tablel LEFT| RIGHT| FULL OUTER JOIN table2 
ON conditions; 
其 中 : 
口 tablel: 数据 表 1。 通 常 在 外 连接 中 被 称 为 左 表 。 
口 table2: 数据 表 2。 通 常 在 外 连接 中 被 称 为 右 表 。 
口 LEFT OUTER JOIN: 左 外 连接 。 使 用 左 外 连接 时 得 到 的 查询 结果 中 ， 除 了 符合 条 
件 的 查询 结果 部 分 ， 还 要 加 上 左 表 中 余下 的 数据 。 

口 RIGHT OUTER JOIN: 右 外 连接 。 使 用 右 外 连接 时 得 到 的 查询 结果 中 ， 除 了 符合 
条 件 的 查询 结果 部 分 ， 还 要 加 上 右 表 中 余下 的 数据 。 

口 FULL OUTER JOIN: 全 连接 。 使 用 全 外 连接 时 得 到 的 查询 结果 中 ， 除 了 符合 条 件 
的 查询 结果 部 分 ， 还 要 加 上 左 表 和 右 表 中 余下 的 数据 。 

口 ON: 设置 外 连接 中 的 条 件 。 与 WHERE 子 句 后 面 的 写法 一 样 。 

下 面 通过 示例 11 来 分 别 演示 左 外 连接 、 右 外 连接 以 及 全 外 连接 的 使 用 。 

【示例 11】 分 别 使 用 3 种 外 连接 查询 节目 信息 表 和 节目 类 型 信息 表 。 

由 于 节目 信息 表 和 节目 类 型 信息 表 是 通过 节目 类 型 编号 字段 关联 的 ， 因 此 ， 可 以 将 两 
张 表 中 节目 类 型 编号 相等 作为 查询 条 件 。 为 了 能 够 更 好 地 看 出 3 种 外 连接 的 区 别 ， 首 先 将 
两 张 数 据 表 中 ， 根 据 节目 类 型 编号 相等 作为 条 件 时 的 记录 查询 出 来 。 语 句 如 下 : 

USE chapter7; 

SELECT * FROM programinfo,typeinfo 

WHERE programinfo.programtype=typeinfo.id; 

执行 上 面 的 语句 ， 效 果 如 图 7.12 所 示 。 

从 图 7.12 所 示 的 查询 结果 可 以 看 出 , 在 查询 结果 左 侧 是 节目 信息 表 中 符合 条 件 的 全 部 
数据 ， 右 侧 是 节目 类 型 信息 表 中 符合 条 件 的 全 部 数据 。 下 面 就 分 别 使 用 3 种 外 连接 来 根据 
示例 11 的 条 件 查询 数据 ， 请 读者 注意 观察 查询 效果 。 

(1) 使 用 左 外 连接 查询 

使 用 左 外 连接 查询 , 将 节目 信息 表 作 为 左 表 , 节目 类 型 信息 表 作 为 右 表 。 查询 语句 如 下 : 

USE chapter77 


SELECT * FROM programinfo LEFT OUTER JOIN typeinfo 
ON programinfo.programtype=typeinfo.id; 
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=|glx 
文件 引 编辑 外 视图 多 查询 @) 项 目 @)， 调式 W) 工具 CD) 窗口 思 社区 已 

者 助 四 

辽 新 娃 查 询 m | 出 | 也 也 本 记忆 加 本 又 下 

TIERRE "| ?3 hv 江 鱼 加 | 站 绚 | 妆 疾 久 名 


SQLQueryl. s... ator (53))*| WIIN-9527WN .prograninfo | 
日 05E chapter7; 


-| YIDY-9527\WAdninistrate .，| chapterT |00:00:00 |7 行 
就 结 行 5 列 1 Chl Ins 


图 7.12 满足 等 值 条 件 的 所 有 记录 
执行 上 面 的 语句 ， 就 可 以 完成 左 外 连接 的 查询 了 。 执 行 效果 如 图 7.13 所 示 。 


rosoft SQL Server Nanagenent Studio 


E =l9lxl 
文件 如 ”编辑 也) 视图 查询 Q) 项 目 @) 调 坛 ) 工具 C) 窗口 电 ， 社区 
帮助 0 
也 .新建 查询 四 | 书 | 迅 记 串 | 刘 | 芒 回电 加 
Me Wo | chspter7 -| ?执行 QW 有 v 叹 部 国 /3” 囊 | 的 


"SQLQueryl. s...ator (53))*| 人 TIW-952TW prograninfo | 
日 0SE chapter7; 
SELECT * FROM programinfo LEFT OUTER JOIN typeinfo 

programinfo.programt ype=typeinfo. id; 


.| MT-952T\WAdninistrate . .| chapter7 |00:00:00 |8 行 
行 5 列 1 Ch 1 


图 7.13 左 外 连接 的 使 用 


Ins 大 


从 图 7.13 中 可 以 看 出 ,最 后 一 条 记录 就 是 在 节目 信息 中 的 信息 ,而 在 类 型 信息 表 中 没 
有 对 应 的 数据 。 由 于 右 表 中 没有 与 之 对 应 的 数据 ， 因 此 所 有 的 数据 全 部 都 用 NULL 代替 。 

(2) 使 用 右 外 连接 查询 

将 节目 信息 表 作为 左 表 ， 节 目 类 型 信息 表 作为 右 表 。 查 询 语句 如 下 : 

USE chapter7; 


SELECT * FROM programinfo RIGHT OUTER JOIN typeinfo 
ON programinfo.programtype=typeinfo.id; 


执行 上 面 的 语句 ， 就 可 以 完成 右 外 连接 的 查询 了 。 执 行 效果 如 图 7.14 所 示 。 
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aIxl 
文件 四 ”编辑 如 ”视图 查询 @) 项 目 @) 调式 W) 工具 C) 窗口 人 ) 社区 加 ) 

才 助 0 

了 着 | 六 | 巴 访 吕 | 让 | 蕊 回电 | 强 s 

: Wy 9 | chepter7 =| ?执行 oO 有 v 器 宇 国政 绚 | 的 关 扩 避 


SQLQueryl. s- - - ator (53))*| WIDN-9527\N .proaraninfo | =x 
日 USE chapter7; 


二 
SELECT * FRON programinfo RIGHT OUTER JOIN typeinfo 
ON programinfo.programtype=typeinfo. id; 


就 绪 行 5 列 1 Chl Ins // 


图 7.14 右 外 连接 的 使 用 


从 图 7.14 所 示 的 结果 可 以 看 出 ,第 7 条 记录 是 左 表 节 目 信 息 表 中 不 存在 的 数据 了 。 由 
于 左 表 中 没有 与 之 对 应 的 数据 ， 因 此 所 有 的 数据 全 部 都 用 NULL 代替 。 

(3) 使 用 全 外 连接 

将 节目 信息 表 作为 左 表 ， 节 目 类 型 信息 表 作为 右 表 。 查 询 语句 如 下 : 

USE chapter7; 


SELECT * FROM programinfo FULL OUTER JOIN typeinfo 
ON programinfo.programtype=typeinfo.id; 


执行 上 面 的 语句 ， 就 可 以 完成 全 外 连接 的 查询 了 。 执 行 效果 如 图 7.15 所 示 。 


soft SQL Server Wanagenent Studio =|DIxl 
文件 中) 编辑 到 ) 视图 GD) 查询 @) 项 目 @) 调试 @) 工具 C) 窗口 @) 社区 C) 帮助 00) 

;也 新建 查询 思 | 出 | 六 讽 加 | 记 | 良 加 局 | 双 扎 

| ehoterT -| 1 执行 Wb》 mm v 中 训 加 | 由 鸡 | 的 圈 入 | 三 名 

SQLqueryl. s...ator (53))*| WOW-S527WN prograninfo | x 
USE chapter7; 


SELECT * FROM programinfo FULL OUTER JOIN typeinfo 
上 oN programinfo.programtype=typeinfo. id; 


加 二 和 已 … | WID-9527 RSSQLSERYER2008 〔 . ，| TDY-952T\WAdninistrate .| chapterT |00:00:00 |9 行 
就 结 行 5 列 1 hl 2 


图 7.15 全 外 连接 的 使 用 


*165* 


第 2 篇 ， 表 操作 基础 


从 图 7.15 所 示 的 结果 中 可 以 看 出 ， 结 果 是 左 表 和 右 表 中 全 部 的 记录 。 
通过 上 面 的 例子 ， 相 信 读 者 已 经 对 外 连接 的 3 种 连接 方式 有 所 了 解 了 。 在 以 后 的 应 用 
中 ， 就 可 以 选择 所 需 的 连接 方式 完成 相关 的 操作 了 。 


7.3.4 ”只 查询 出 符合 条 件 的 数据 一 一 内 连接 


与 外 连接 对 应 的 是 内 连接 ， 内 连接 可 与 外 连接 截然 不 同 了 。 内 连接 可 以 理解 成 是 等 值 
连接 ， 也 就 是 说 查询 的 结果 全 部 都 是 符合 条 件 的 数据 。 但 是 ， 内 连接 的 语法 形式 与 外 连接 
很 相似 。 有 具体 的 语法 形式 如 下 : 


SELECT column namel, column name2,...... 

FROM tablel INNER JOIN table2 

ON conditions 

其 中 ， 

口 tablel: 数据 表 1。 通 常 在 外 连接 中 被 称 为 左 表 。 

口 table2: 数据 表 2。 通 常 在 外 连接 中 被 称 为 右 表 。 

口 INNER JOIN: 内 连接 的 关键 字 。 

口 ON: 设置 内 连接 中 的 条 件 。 与 外 连接 中 的 ON 关键 字 是 一 样 的 。 

下 面 就 用 示例 12 来 演示 如 何 使 用 内 连接 。 

【示例 12】 使 用 内 连接 查询 节目 信息 表 和 节目 类 型 信息 表 。 

在 使 用 内 连接 查询 时 ， 使 用 的 条 件 仍然 是 节目 信息 表 和 节目 类 型 信息 表 中 的 节目 编号 
相等 。 查 询 语句 如 下 : 

USE chapter7; 

SELECT * FROM programinfo INNER JOIN typeinfo 

ON programinfo.programtype=typeinfo.id; 

执行 上 面 的 语句 ， 就 可 以 完成 内 连接 查询 了 。 执 行 效果 如 图 7.16 所 示 。 
=Iolxl 


ver Wanagenent Studi 


ET rT ET 查询 @) 项 目 中 调式 @) 工具 上) 窗口 如 社区 人 帮助 了 D 
习 新 可 询 WD | 书 | 僻 计 咏 | 市 | 蕊 回 结 疫 s 
sc -| ?执行 0 有 品 辐 国 | 下 i 


-ator (53))*| WDN-9527\WN, .prograninfo | 
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行 5 列 1 Ch 1 


图 7.16 内 连接 的 使 用 
从 图 7.16 中 可 以 看 出 ， 内 连接 查询 的 结果 就 是 符合 条 件 的 全 部 数据 。 
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7.4 结果 集 的 运算 


前 面 讲 的 全 都 是 查询 语句 的 写法 ， 执 行 完 每 一 个 查询 语句 后 ， 都 会 得 到 一 个 结果 集 。 
那么 ， 可 以 一 次 查看 多 个 结果 集 吗 ? 当然 是 可 以 的 ， 在 本 节 中 就 带 读者 来 体会 如 何 操作 结 
果 集 。 


7.4.1 ”使 用 UNION 关键 字 合并 查询 结果 


所 谓 合并 查询 结果 就 是 将 两 个 或 更 多 的 查询 结果 放 到 一 个 结果 集中 显示 ， 但 是 合并 结 
果 是 有 条 件 的 , 那 就 是 必须 要 保证 每 一 个 结果 集中 的 字段 和 数据 类 型 一 致 。UNION 关键 字 
就 是 用 于 合并 多 个 结果 集 的 ， 具 体 的 语法 如 下 : 

SELECT column namel, column name2,..FROM table namel 

UNION [ALL] 


SELECT column namel, column name2,..FROM table namel; 
UNION 


ORDER BY column name 


其 中 : 

口 UNION: 合并 查询 结果 的 关键 字 。 结 果 中 会 去 掉 相 同 的 行 。 

口 UNION ALL: 与 UNION 类 似 ， 但 是 在 结果 中 不 会 去 掉 重 复 的 行 。 

口 ORDER BY: 对 结果 集 进行 排序 。 在 对 结果 集 进行 排序 时 ， 是 对 第 一 个 查询 中 的 

字段 进行 排序 的 。 

下 面 就 用 示例 13 来 演示 如 何 使 用 UNION 合并 查询 结果 。 

【示例 13】 查 询 节 目 信 息 表 中 的 节目 编号 和 节目 名 称 以 及 节目 类 型 信息 表 中 的 类 型 编 
号 与 类 型 名 称 ， 并 将 两 个 查询 结果 使 用 UNION 关键 字 合 并 。 

根据 题目 要 求 ， 也 就 是 完成 两 个 查询 语句 ， 并 将 两 个 查询 用 UNION 关键 字 连 接 。 碍 
询 语 句 如 下 : 

USE chapter7; 

SELECT id, name FROM programinfo 

UNION 

SELECT id, name FROM typeinfo; 

执行 上 面 的 语句 ,就 可 以 将 两 个 查询 结果 合并 成 一 个 查询 结果 集 了 。 执 行 效 果 如 图 7.17 
所 示 。 

从 图 7.17 所 示 的 查询 结果 中 可 以 看 出 , 将 节目 信息 表 和 节目 类 型 信息 表 中 的 数据 都 合 
并 到 一 个 结果 集中 了 。 另 外 ， 在 查询 结果 中 可 以 按照 合并 后 的 结果 升序 排列 。 


7.4.2 排序 合并 查询 的 结果 


在 示例 13 中 ， 读 者 已 经 看 到 了 查询 结果 默认 的 排序 方式 是 升序 排列 。 那 么 ， 能 够 改 
变 这 种 排序 方式 吗 ? 当然 可 以 了 ， 在 前 面 的 合并 查询 结果 集 语法 中 就 有 ORDER BY 子 句 。 
也 就 是 说 ， 可 以 使 用 ORDER BY 子 句 对 合并 后 的 结果 集 进行 排序 。 下 面 就 将 示例 13 的 查 
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询 结果 进行 排序 。 

【示例 14】 将 示例 13 的 结果 按照 id 列 进行 降序 排列 。 

降序 排列 使 用 的 是 DESC 关键 字 ， 具 体 的 查询 语句 如 下 : 

USE chapter7; 

SELECT id, name FROM programinfo 

UNION 

SELECT id, name FROM typeinfo 

ORDER BY id DESC; 

执行 上 面 的 语句 ， 就 可 以 将 查询 结果 按照 编号 列 id 降序 排列 。 执 行 效 果 如 图 7.18 
所 示 。 
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图 7.17 使 用 UNION 关键 字 合 并 查询 结果 图 7.18 使 用 ORDER BY 对 结果 集 排序 
从 图 7.18 中 可 以 看 出 ， 查 询 结果 确实 是 按照 id 进行 了 降序 排列 。 
7.4.3 使 用 EXCEPT 关键 字 对 结果 集 差 运算 


结果 集 不 仅 可 以 进行 合并 运算 ， 也 可 以 进行 差 运算 。 差 运算 并 不 是 简单 地 对 结果 集 的 
内 容 进行 减法 运算 ， 而 是 从 一 个 结果 集中 去 除 另 一 个 结果 集中 的 内 容 。EXCEPT 关键 字 的 
法 与 UNION 类 似 ， 语 法 如 下 : 

SELECT column namel, column name2,..FROM table namel 

EXCEPT 


SELECT column namel, column name2,...FROM table namel; 
EXCEPT 


ORDER BY column name 


这 里 ，EXCEPT 就 是 连接 结果 集 之 间 的 关键 字 ， 用 于 集合 差 值 的 运算 。 读 者 可 能 还 会 
犯 糊涂 ， 不 要 紧 ! 通过 下 面 的 示例 15 你 就 可 以 完全 清楚 它 的 用 法 了 。 


"168 


第 7 章 查询 语句 提高 


【示例 15】 查询 节目 信息 表 中 的 信息 。 使 用 结果 集运 算 去 除 所 有 歌舞 类 节目 。 

如 果 要 去 除 所 有 歌舞 类 节目 ,假设 已经 知道 了 歌舞 类 节目 的 类 型 编号 是 1， 那 么 ， 就 
可 以 将 该 查询 作为 EXCEPT 后 面 的 查询 了 。 查 询 语句 如 下 : 

USE chapter7; 


SELECT* FROM programinfo 
EXCEPT 


SELECT* FROM programinfo WHERE programtype=1; 
执行 上 面 的 语句 ， 就 可 以 得 到 第 1 个 查询 结果 去 除 第 2 个 查询 结果 的 值 。 执 行 效 果 如 
图 7.19 所 示 。 


=9Ix| 
文件 外 ”编辑 于 视图 查询 @) 项 目 P) 调试 @) 工具 CD) 窗口 中 
社区 人 C) 才 助 员 
ET 
WY Wd | chepter7 -| 执 和 SW ha v 中 加 国 | 中 各 
SQLQueryl. s... ator (53))*| WIMN-0527\WN proganinfo | 号 X 
日 USE chapter7: 


月 SELECT* FRON programinfo 
| EXCEPT 
FSELECT* FRON programinfo WHERE programtype=1; 


刘 青 

壬 菲菲 ”无 

章 欢 无 

陈 新 无 

王 明 史 口技 


fi27\MSSQLSERYER2008 〔 . ，| WIDW-9527\Adninistrato. .. | chapterT | 00:00:00 | 5 行 
就 绪 行 7 列 1 Ch 1 


阿 


图 7.19 使 用 EXCEPT 对 结果 集 进 行 差 运算 
从 图 7.19 所 示 的 结果 中 可 以 看 出 ， 已 经 不 含有 节目 类 型 编号 (programtype) 是 1 的 节 
目 信息 了 。 但 是 ， 请 读者 一 定 要 记 住 ， 进 行 差 集运 算 时 也 要 保证 EXCEPT 前 后 的 两 个 结果 
集 列 的 个 数 和 数据 类 型 一 致 。 
7.4.4 使 用 INTERSECT 关键 字 对 结果 集 交 运算 


结果 集 除了 合并 和 求 差 运算 外 还 有 一 种 比较 常用 的 运算 ， 那 就 是 取 交 集 。 交 集 这 个 概 
念 对 于 读者 来 说 已 经 不 陌生 了 ， 也 就 是 取 两 个 结果 集中 的 公共 部 分 鹃 。 对 结果 集 取 交集 使 
j INTERSECT 关键 字 ， 它 的 语法 形式 也 与 前 面 的 合并 、 求 差 运算 类 似 ， 具 体 的 语法 形式 
如 下 : 


SELECT column namel, column name2..FROM table namel 
INTERSECT 


SELECT column namel, column name2..FROM table namel; 
INTERSECT 


ORDER BY column name 


a 
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这 里 ，INTERSECT 就 是 用 来 连接 结果 集 求 交集 的 。 有 了 前 面 对 集 合 操作 的 基础 ， 交 
集运 算 就 容易 得 多 了 。 

【示例 16】 查询 节目 信息 表 。 并 使 用 INTERSECT 关键 字 得 到 所 有 歌舞 类 的 节目 
信息 。 

要 通过 INTERSECT 关键 字 得 到 歌舞 类 的 节目 信息 ， 也 就 是 说 歌舞 类 节目 就 是 得 到 的 
交集 了 。 那 么 ，INTERSECT 后 面 的 查询 就 应 该 查 到 所 有 歌舞 类 节目 信息 。 这 里 ， 歌 舞 类 
节目 的 类 型 编号 也 是 已 知 的 ， 即 为 1。 查 询 语句 如 下 : 

USE chapter7; 

SELECTx FROM programinfo 


INTERSECT 
SELECT* FROM programinfo WHERE programtype=1; 


执行 上 面 的 语句 ， 即 可 取得 两 个 结果 集中 的 交集 。 执 行 效果 如 图 7.20 所 示 。 


microsoft SQL Server Nanagenent Studio =|DIxl 
文件 下) 编辑 也 ) 视图 YW) 查询 @) 项 目 E) 调式 @) 工具 CD) 窗口 岂 ) 
社区 CC) 帮助 由 
;也 .新 于 查 询 | 由 | 轧 也 台 | 记 | 功 加 本 | 过 下 
Wy 2 | chepterr -| 执行) >》 入 鸣 辕 | 国 | 中 
SQLQueryl. s... ator (53))*| TD-S52TUWL prograninfo | sx 
章 0SE chapcer7: 二 
由 SELECT* FRON programinfo 
INTERSECT 
SELECT* FROM programinfo WHERE programtype=1; = 


li27 SSQLSERVER2008 〔 . ，| WIIY-9527\WAdninistrate .. | chapterT | 00:00:00 | 3 行 
就 绪 行 6 列 1 Ch 1 有 


图 7.20 使 用 INTERSECT 关键 字 取 得 结果 集 的 交集 


从 图 7.20 所 示 的 结果 可 以 看 出 ， 确 实 是 只 查询 出 了 节目 类 型 (programtype) 是 1 的 节 
目 信 息 。 


全 说 明 : 集合 运算 符 UNION、EXCEPT 和 INTERSECT 也 是 有 优先 级 的 ，INTERSECT 的 
优先 级 是 最 高 的 。 其 余 的 UNION、EXCEPT 优先 级 是 一 样 的 ， 谁 先 在 前 面 就 先 
执行 谁 。 当 然 ， 优 先 级 也 可 以 通过 加 括号 进行 控制 。 


7.5 本 章 小 结 


通过 本 章 的 学 习 ， 读 者 应 该 对 查询 语句 有 了 更 进一步 的 了 解 了 。 在 本 章 中 主要 带领 读 
者 学 习 了 几 种 在 实际 应 用 中 经 常 使 用 的 查询 语句 ， 主 要 包括 子 查询 、 分 组 查询 、 多 表 查 询 
以 及 如 何 对 结果 集 进行 运算 。 这 几 种 查询 语句 并 不 是 独立 存在 的 ， 可 以 根据 实际 情况 综合 
选择 使 用 查询 语句 ， 也 就 是 说 ， 在 一 个 查询 语句 中 可 以 是 多 种 查询 语句 并 存 。 


一 、 填 空 题 


1. 子 查询 中 常用 的 关键 字 有 。 
2. 外 连接 的 形式 有 6 
3. 合并 查询 结果 的 关键 字 是 6 
二 、 选 择 题 
1. 判断 某 一 个 查询 语句 是 否 能 够 查询 出 结果 使 用 的 关键 字 是 
A. IN B. NOT C. EXISTS D. 以 上 都 不 对 
2. 下 面 对 子 查询 的 描述 正确 的 是 8 
A. 子 查询 就 是 在 一 个 查询 中 包含 另 一 个 查询 
B. 子 查询 只 能 返回 一 个 值 
C. 子 查询 只 能 返回 多 个 值 
D. 以 上 都 不 对 
3. 下 面 对 多 表 查 询 的 描述 正确 的 是 
A. 如 果 在 多 表 查 询 时 没有 使 用 WHERE 条 件 ， 则 会 出 现 稍 卡 尔 积 
B. 同一 个 表 之 间 的 连接 称 为 自 连接 
C. 多 表 查 询 分 为 内 连接 、 外 连接 以 及 自 连接 
D. 以 上 都 对 
、 问 答题 


1. 在 什么 情况 下 选择 IN 关键 字 ? 
2. 什么 是 分 组 查询 ? 分 组 查询 使 用 的 关键 字 是 什么 ? 
3. 结果 集运 算 有 什么 作用 ? 


ul 


a 
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在 前 面 的 章节 中 曾 学 过 聚合 函数 ， 它 就 是 一 类 系统 函数 。 所 谓 系统 函数 ， 可 以 理解 成 
是 安装 SQL Server 后 就 有 的 函数 ， 可 以 直接 使 用 。 实 际 上 , 在 SQL Server 中 除了 系统 函数 
之 外 还 有 用 户 自 定义 函数 。 自 定义 函数 有 什么 用 呢 ? 这 就 好 像 是 去 购买 一 台电 脑 ， 但 是 所 
有 品牌 的 电脑 都 满足 不 了 你 的 需求 ， 那 就 只 有 去 DIY 一 台 适 合 你 的 电脑 了 。 当 系统 函数 满 
足 不 了 需求 时 就 可 以 考虑 自己 定义 一 个 函数 使 用 。 

本 章 的 主要 知识 点 如 下 : 

口 了 解 和 使 用 系统 函数 

口 如 何 定义 和 使 用 自 定义 函数 


8.1 系统 函数 


系统 函数 给 我 们 带 来 了 便利 的 工具 ， 要 想 正确 使 用 这 个 工具 就 要 知道 系统 函数 都 有 哪 
些 。 在 SQL Server 中 ， 系 统 函 数 主 要 分 为 数学 函数 、 字 符 串 函数 、 日 期 和 时 间 函 数 等 。 在 
本 节 中 就 将 带领 读者 认识 这 些 系统 函数 。 
8.1.1 数学 函数 

所 谓 数学 函数 ， 就 是 对 数值 类 型 字段 的 值 进行 运算 的 函数 。 数 学 函数 对 读者 来 说 ， 应 
该 是 最 熟悉 不 过 的 了 ， 因 为 从 小 学 就 开始 接触 数学 运算 了 。 那 么 ， 在 SQL Server 数据 库 中 
数学 函数 包括 哪些 呢 ? 主 要 包括 了 取 绝 对 值 函数 、 取 正弦 函数 、 取 余弦 函数 、 取 正切 函数 
以 及 取 对 数 函 数 等 。 为 了 能 够 让 读者 对 数学 函数 有 一 个 全 面 的 了 解 ， 下 面 将 常用 的 数学 函 
数列 在 表 8-1 中 讲解 。 

表 8-1 数学 函数 说 明 


序号 函数 形式 说 明 
取 x 的 绝对 值 函 数 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 。 当 输入 的 
1 ABS(x) 参数 是 整数 时 ， 返 回 值 就 是 该 数 本 身 ， 当 输入 的 参数 是 负数 时 ， 返 回 值 就 


是 去 掉 负 号 后 的 数值 ，0 取 绝 对 值 还 是 0 
取 x 的 指数 函数 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 。 返回 x 的 指 
数值 ， 也 就 是 ex 
3 POWER(%,y) | 取 x 的 y 次 究 。 该 函数 有 两 个 参数 ， 参 数 类 型 都 可 以 是 float 类 型 的 
按照 指定 精度 y 对 x 四 舍 五 入 。 该 函数 有 两 个 参数 ，x 是 用 来 进行 四 舍 五 
入 的 参数 ， 类 型 是 float; y 是 精度 ， 类 型 是 int 

5 SQRT(x) 取 x 的 平方 根 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 


2 EXP(x) 


4 ROUND(x,y) 
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序号 函数 形式 说 明 

6 SQUARE(x) “| 取 x 的 平方 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 

7 PIO i 

8 FLOOR(x) 取 小 于 x 的 最 小 整数 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
9 CEILING(x) 取 大 于 x 的 最 大 整数 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
10 LOG(x) 取 x 的 自然 对 数 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 

11 LOGI10(x) 取 x 的 以 10 为 底 的 对 数 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
忠 SIN(x) 取 x 的 三 角 正 弦 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
13 COS(x) 取 x 的 三 角 余 弦 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
14 TAN(x) 取 x 的 三 角 正 切 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
15 COT(%) 取 x 的 三 角 余 切 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 
16 ASIN(x) 取 x 的 反正 弦 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 

17 ACOS(x) 取 x 的 反 余弦 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 

18 ATAN(x) 取 x 的 反正 切 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 

19 ACOT(x) 取 x 的 反 余 切 值 。 该 函数 只 有 一 个 参数 ， 参 数 是 float 类 型 的 


从 表 8-1 中 可 以 看 4 


函数 。 无 论 是 哪 种 函数 ， 


只 要 


1， 前 11 个 函数 都 是 一 般 的 数值 运算 函数 ; 后 面 的 8 个 函数 是 三 角 


读者 能 够 正确 地 找到 要 使 用 的 函数 ， 并 传 入 函数 要 求 的 参数 ， 


就 可 以 很 好 地 使 用 这 个 函数 了 。 下 面 分 别 通过 示例 来 演示 如 何 使 用 数值 运算 函数 和 三 角 


函数 。 


【示例 1】 按照 下 列 要 求 使 用 数值 运算 函数 进行 计算 。 

(1) 使 用 函数 计算 5 的 平方 以 及 36 的 平方 根 。 

(2) 使 用 函数 计算 半径 是 3 的 圆 面 积 。 

(3) 使 用 函数 计算 3 的 4 次 暴 。 

(4) 使 用 函数 取 5.28 的 最 大 整数 和 最 小 整数 。 

要 使 用 的 是 数学 函数 中 的 数值 运算 函数 ， 现 在 就 得 要 求 读者 根据 题目 的 要 求 在 表 8-1 
中 找到 适合 的 函数 来 计算 了 。 

(1) 计算 平方 的 函数 是 SQUARE， 计 算 平方 根 的 函数 是 SQRT， 语 句 如 下 

SELECT SQUARE (5), SQRT (36); 


执行 上 面 的 语句 ， 就 可 以 得 到 5 的 平方 和 36 的 平方 根 了 。 执 行 效果 如 图 8.1 所 示 。 这 
里 ， 要 提醒 读者 如 果 只 是 用 SELECT 语句 而 不 使 用 数据 表 ， 就 不 用 指定 数据 库 了 。 

从 图 8.1 所 示 的 结果 中 可 以 看 出 ， 通 过 函数 计算 后 5 的 平方 是 25，36 的 平方 根 是 6。 

(2) 计算 圆 的 面积 ， 需 要 知道 圆 的 半径 和 PI 的 值 ，PI 的 值 可 以 通过 PI 函数 来 得 到 。 


语句 如 下 : 


SELECT PI ()*SQUARE (3); 


执行 上 面 的 语句 ， 就 可 以 得 到 半径 是 3 的 圆 面积 。 执 行 效果 如 图 8.2 所 示 。 
从 图 8.2 所 示 的 结果 可 以 看 出 ， 半 径 是 3 的 圆 面积 是 28.2743338823081。 


sr 
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Ricrosoft SQL Server Bensese0 Se 
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图 8.1 SQUARE0 和 SQRTO 函 数 的 使 用 图 8.2 PIO 函数 的 使 用 

(3) 计算 x 的 y 次 突 使 用 的 函数 是 POWER。 语 句 如 下 : 

SELECT POWER (3, 4); 

执行 上 面 的 语句 ， 就 可 以 得 到 3 的 4 次 容 的 计算 结果 了 。 效 果 如 图 8.3 所 示 。 

从 图 8.3 的 结果 可 以 看 出 ，3 的 4 次 容 的 结果 是 81。 

(4) 取 最 大 整数 用 函数 FLOOR 函数 ， 取 最 小 整数 用 函数 CEILING。 语 句 如 下 : 
SELECT FLOOR (5.28), CEILING (5.28); 

执行 上 面 的 语句 ， 就 可 以 得 到 5.28 的 最 大 整数 和 最 小 整数 值 了 。 效 果 如 图 8.4 所 示 。 
从 图 8.4 所 示 的 结果 可 以 看 出 ，5.28 取 最 小 整数 是 S， 最 大 整数 是 6。 
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SQLQuery1. s... ator (53))* x ee, 
4 二 ry ker 和 A 


SELECT FLOOR (5.28), CEILING (5.28); 


Pdninistrate .. | chapter7 |00:00:00 | 1 行 
就 结 行列 Ch 


图 8.3 POWER() 函 数 的 使 用 图 8.4 FLOORO 函 数 和 CEILINGO) 函 数 的 使 用 


【示例 2】 按照 下 列 要 求 使 用 三 角 函 数 计 算 值 。 

(1) 使 用 三 角 函 数 计算 0.5 的 正弦 值 和 余弦 值 。 

(2) 使 用 三 角 函 数 计算 0.8 的 正切 值 和 余 切 值 。 

(3) 使 用 三 角 函 数 计 算 0.6 的 反正 弦 值 和 反正 切 值 。 

下 面 就 要 根据 不 同 的 要 求 ， 选 择 表 8-1 中 不 同 的 三 角 函 数 。 读 者 可 以 先 自己 选择 三 角 
函数 计算 出 结果 ， 再 与 下 面 的 答案 进行 对 照 。 

(1) 取 正 弦 值 的 函数 是 SINO， 取 余弦 值 的 函数 是 COS()。 语 句 如 下 : 


SELECT SIN (0.5), COS (0.5); 
执行 上 面 的 语句 ， 即 可 计算 出 0.5 的 正弦 值 和 余弦 值 。 效 果 如 图 8.5 所 示 。 


.174 。 


C.. | TDW-9527\WAdninistrate . .| chapter7 | 00:00:00 |1 行 
就 结 TREE 
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从 图 8.5 所 示 的 结果 可 以 看 出 ，0.5 取 正 弦 值 的 结果 是 0.48 (保留 2 位 小 数 )， 余 弦 值 
是 0.88〔〈 保 留 2 位 小 数 )。 
(2) 取 正 切 值 使 用 TAN， 取 余 切 值 使 用 COT。 语 句 如 下 


SELECT TAN (0.8), COT (0.8) 


从 图 8.6 所 示 的 结果 可 以 看 出 ，0.8 的 正切 值 是 1.03 (保留 2 位 小 数 )， 余 切 值 是 0.97 
(保留 2 位 小 数 )。 


了 6 加 a 
文件 下) 编辑 EE) 视图 @) 查询 @) 项 目 EE) 文件 时) ”编辑 下 ) 视图) 查询 @) 项目 到 ) 
调试 四 ) 工具 XY) 窗口 和 社区 C) 帮助 0 调试 @) 工具 CO) 窗口 和 社区 CC) 帮助 0 
卫 新建 查 询 | 由 | 也 也 加 | 记 | 七 加 电 卫 新 建 查询 mg | 让 也 也 可 | 应 | 蕊 加 名 
Wy Hy | chapter7 ”| 时 执 行 &) 名 YH3 | chapter? >| 执行 &) 总 


‘SQLQueryl. s... ator (53))* vx 
日 SELECT TAN (0.8)}, COT (0.8); 


SQLQueryl. s... ator (53))* 
日 SELECT SIN(0.5),cCOS(0.5 


加 结果 | 消息 | 


| 102963855705036 | 0.971214600650474 


TDY-9527\Adninistrate . .| chapter7 | 00:00:00 |1 行 [TDW-9527\Adninistrato. . ,| ehapter7 |00:00:00 |1 行 
就 结 行 3 列 1 Ch1 上 就 绪 行列 1 Ch1 2 


图 8.5 ”SINO 函 数 和 COS() 函 数 的 使 用 图 8.6 ”TAN(O) 函 数 和 COTO 函 数 的 使 用 


(3) 取 反 正弦 值 的 函数 是 ASIN， 取 反正 切 值 的 函数 是 ATAN。 语 句 如 下 
SELECT ASIN (0.6), ATAN (0.6); 
执行 上 面 的 语句 ， 就 可 以 得 到 0.6 的 反正 弦 值 和 反正 切 值 了 。 效 果 如 图 8.7 所 示 。 
从 图 8.7 所 示 的 结果 可 以 看 出 ，0.6 的 反正 弦 值 

是 0.64 (保留 2 位 小 数 )， 反 正切 值 是 0.54 (保留 2 lie 


文件 人 编辑 下 ) 视图 WV) 查询 @@) 项 目 E) 


位 小 数 )。 调 荆 史 ” 工 具 中 窗口) 社区 和 助人 0 
了 外 mg | 记 | 也 也 四 | 记 | 芝 加 总 

[2 ~ 米 | Wy 3 | -hapter7 | 执行 &) 旦 

8.1.2 字符 串 函 数 SQLOueryl. s... ator (53))* -x 


有 了 前 面 的 数学 函数 基础 读者 应 该 已 经 明 B 了 | 

函数 是 如 何 使 用 的 了 。 在 实际 应 用 中 ， 除 了 对 数值 类 
型 的 人 操作 需要 函数 之 外 , 对 字符 串 类 型 的 数据 也 同 [RCI 
样 需要 函数 字符 叫 函数 主要 包括 将 字符 品 转换 成 大 。 呈 ET 
写 、 将 字符 串 转 换 成 小 写 、 截 取 字符 串 中 某 些 字符 以 “图 87 ASINO 和 函数 和 ATANO 函 数 的 使 用 
及 向 字符 串 中 插入 字符 等 .常用 的 字符 叫 函数 名 称 和 


器 结果 | 2 消息 | 
广 jEaal | 


人 | 0543501108793284 | 0.540419500270584 


使 用 方法 如 表 8-2 所 示 。 
表 8-2 ”字符 串 函数 
函数 形式 说 明 
ee 用 于 取 x 的 ASCII 值 。 该 函数 只 有 一 个 参数 ， 并 且 这 个 参数 不 仅 可 以 是 
全 -个 字符 串 ， 也 可 以 是 一 个 表达 式 


取 字 符 串 x 中 从 y 处 开始 的 z 个 字符 。 该 函数 有 3 个 参数 ，x 代表 字符 
SUBSTRING(x,y,z) | 串 或 表达 式 ，y 代表 从 哪个 位 置 开 始 截取 字符 串 ，z 代表 取 几 个 字符 。 这 
里 ，y 和 z 都 是 整数 类 型 


es 
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续 表 
序号 函数 形式 说 有明 
取 字 符 串 y 中 指定 表达 式 x 的 开始 位 置 。 该 函数 有 2 个 参数 ，x 代表 的 是 
3 | CHARINDEXQy) | 要 查找 的 字符 串 ，y 代表 的 是 指定 的 字符 串 
ee 取 字 符 串 x 中 从 左边 开始 指定 个 数 y 的 字符 。 该 函数 有 2 个 参数 ，x 代表 
Cy) 的 是 一 个 给 定 的 字符 串 ，y 代表 取 字 符 串 的 个 数 。 这 里 ，y 是 整数 类 型 
I 取 字 符 串 x 中 从 左边 开始 指定 个 数 y 的 字符 。 该 函数 有 2 个 参数 ，x 代表 
Cey) 的 是 一 个 给 定 的 字符 串 ，y 代表 取 字 符 串 的 个 数 。 这 里 ，y 是 整数 类 型 
6 LEN(x) 取 字 符 串 x 的 长 度 。 该 函数 需要 一 个 字符 串 类 型 的 参数 
7 |LOWERG) 将 x 中 的 大 写字 母 转换 成 小 写字 母 。 该 函数 需要 一 个 字符 串 类 型 的 参数 
8 | UPPERG) 将 x 中 的 小 写字 母 转换 成 大 写字 母 。 该 函数 需要 一 个 字符 串 类 型 的 参数 
9 LTRIM(x) 取 x 去 除 第 一 个 字符 前 空格 后 的 字符 串 。 该 函数 需要 一 个 字符 串 类 型 的 参数 
去 除 最 后 一 个 字符 前 空格 后 的 字符 串 。 该 函数 需要 一 个 字符 串 类 型 的 
ey 家 去 除 和 后 个 字符 前 空格 后 的 字符 串 。 该 函数 需要 一 个 字符 串 类 型 的 
换 x 字符 串 中 出 现 的 所 有 y 字符 串 。 该 函数 需要 3 个 字符 串 类 型 
ee 用 普 纪 x 字 符 申 中 出现 的 所 有 字符 该 函数 需要 3 个 字符 串 类 型 的 
12 | REVERSE(x) 取得 x 字符 串 逆序 的 结果 。 该 函数 需要 一 个 字符 串 类 型 的 参数 
13 | SPACE(x) 取得 x 个 空格 组 成 的 字符 串 
读者 看 过 表 8-2 后 ， 会 发 现 字符 串 函数 不 过 就 这 几 个 嘛 ， 很 简单 ， 实 际 上 还 有 很 多 字 
符 串 函数 不 太 常 用 ， 这 里 没有 列 出 。 在 表 8-2 中 列 出 的 13 个 函数 中 ， 有 些 函 数 只 需要 一 个 
参数 , 而 有 些 函数 则 需要 2 个 或 3 个 函数 。 为 了 让 读者 能 够 完成 掌握 表 中 列 出 的 这 些 函 数 ， 


下 面 就 分 别 举例 说 明 这 些 函 数 的 用 法 。 


【示例 3】 


根据 下 面 的 要 求 选择 合适 的 函数 来 实现 。 


(1) 给 定 字符 串 “abcdefga”， 将 其 中 a 换 成 A。 

(2) 给 定 字符 串 “abcdefabcdef”， 计 算 该 字符 串 的 长 度 ， 并 将 其 逆序 输出 。 
(3) 给 定 字 符 串 “abcdefg”， 从 左边 取 该 字符 串 的 前 3 个 字符 。 

(4) 给 定 字符 串 “aabbcc”， 将 该 字符 串 转 换 成 大 写 。 

(5) 给 定 字符 串 “abcdefg”， 查 看 字符 “b” 在 该 字符 串 中 的 位 置 。 

根据 题目 要 求 ， 请 读者 选择 不 同 的 字符 串 函数 ， 完 成 本 题 中 的 5 个 小 题 。 
(1) 蔡 换 字符 串 中 的 字符 ， 可 以 选择 REPLACE 函数 。 语 句 如 下 


SELECT REPLACE ('abcdefga','a','A'); 


执行 上 面 的 语句 ， 就 可 以 将 字符 串 中 的 所 有 的 “a” 换 成 “A” 了 。 效 果 如 图 8.8 所 示 。 
(2) 计算 字符 串 长 度 使 用 的 函数 是 LEN， 逆 序 输出 使 用 的 是 REVERSE 函数 。 语 名 


3 
3 
3 


如 下 : 


SELECT LEN ('abcdefabcdef'), 
执行 上 面 的 语句 ， 就 可 以 完成 取 字符 串 的 长 度 和 逆序 输出 的 效果 。 效 果 如 图 8.9 所 示 。 
从 图 8.9 的 结果 可 以 看 出 ， 该 字符 串 的 长 度 是 12， 逆 序 输出 的 是 “fedcbafrdcba”。 
(3) 从 左边 开始 截取 字符 串 的 函数 是 LEFT。 语 句 如 下 : 


SELECT LEFT ('abcdefg', 3); 


REVERSE ('abcdefabcdef'); 


a 
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i erandk 
Microsoft SQL Server Nanageneatl Se =|D[X| Microsoft SQL 


Server Wanagenent Studio -Iolx| 


文件 编辑 下 视图 四， 查询 @) 项 目 人 E) 调 坛 中) 文件 四 编辑 到 ) 视图) 查询 @) 项目 E) 调 坏 W) 工具 CD) 窗口) 
工具 四 宣 D 思 社区 C) 帮助 中 社区 人 天 助 吕 
;也 新建 查询 名 | 入 | 创 久久 | 昌 | 了 BS3I 妥 5 : 驴 吉 3 询 加 | 六 | 说 计 囊 | 届 | 苞 回 纪 | 内 
;也 级 | easter -| 执行 QW) 》 加 总 时 tr -| 执行) 有 v 品 癌 国 |] 3 
“SQLQueryl. s... ator (54))* ER SQLQueryl. s... ator (54))*| vx 
加 SELECT REPLACE ('abcdefga','a','A'); SELECT LEN ('abcdefabcdef'), REVERSE ('abcdefabcdef'); 去 
器 结果 | 2 消息 | 器 结果 | 2 消息 | 
[| Fans) | 
12 | edcbaledcba 
po08 〔 .，| WIDN-9527\Adninistrato, .. |master |00:00:00 |1 行 Jrow-9527 \nssQLSERYVER2008 (.. | TDY-9527\Adninistrate .. |mester |00:00:00 |1 行 
就 绪 行 3 列 ! Cl 者 就 绪 行 4 列 1 Ch1 > 
图 8.8 REPLACEO 函 数 的 使 用 图 8.9 LENO 函 数 和 REVERSEO 函 数 的 使 用 


执行 上 面 的 语句 ， 就 可 以 取出 该 字符 串 中 的 前 3 个 字符 。 效 果 如 图 8.10 所 示 。 


从 图 8.10 所 示 的 结果 可 以 看 出 ， 取 该 字符 串 中 的 
(4) 将 字符 串 转换 成 大 写 使 用 的 函数 是 UPPER。 


SELECT UPPER ('aabbcc'); 


前 3 个 字符 是 abc。 
语句 如 下 : 


执行 上 面 的 语句 ， 就 可 以 将 该 字符 串 转 换 成 大 写 了 。 效 果 如 图 8.11 所 示 。 
从 图 8.11 所 示 的 结果 可 以 看 出 ， 已 经 将 “aabbcc” 转 换 成 了 “AABBCC ”。 


Ricrosoft SQL Server Nanseenesel oe 
文件 人 ) 编辑 到 ) 视图 WW) 查询 @) 项 目 到 ) 
调试 四 工具 CD) 窗口 和 ) 社区 人 ) 帮助 人 D 
也 .新建 查询 中 | 记 | 也 也 焉 记功 四 名 
RY 2 | naster . 


Ricrosoft SQL Server Neneeone 


=I9lx| 
文件 四 编辑) 视图 GD 查询 @) 项 目 引 ) 
调式 四 ”工具 CO) 窗口 WD 社区 帮助 中 


也 新建 查询 | 由 | 这 也 四 | 包 | 切 加 日 
Wy 32 | naster ~ 站 


“SQLQueryl. s... ator (54))* 
日 SELECT LEFT ('abcdefg', 3); 


Waninistrate . .| master |00:00:00 |1 行 


SQLQueryl. s... ator (54))* vx 
日 SELECT UPPER ('aabbcc'): 


局 辣 已 消息 | 
|1_[ AABBCC 


DW-9527\Adninistrato. .|master |00:00:00 | 1 行 


行 4 列 1 Ch1l 


图 8.10 LEFTO 函 数 的 使 用 


就 绪 行列 1Ch1 
图 8.11 UPPERO 函 数 的 使 用 


(5) 查找 “b” 在 字符 串 “abcdefg” 中 的 位 置 。 语 句 如 下 : 


SELECT CHARINDEX ('b','abcdefg'); 


执行 上 面 的 语句 ， 效 果 如 图 8.12 所 示 。 

从 图 8.12 所 示 的 结果 可 以 看 出 ,“b” 在 字符 串 
“abcdefg” 中 的 位 置 是 2。 

上 面 的 示例 仅 使 用 了 字符 串 函数 中 的 一 部 分 函 
数 ， 有 兴趣 的 读者 可 以 将 表 8-2 中 的 剩余 函数 全 部 演 
练 一 下 。 


8.1.3 日 期 时 间 函 数 


日 期 时 间 函 数 也 是 系统 函数 中 的 一 个 重要 组 成 部 
分 。 使 用 日 期 时 间 函 数 可 以 方便 地 获取 系统 的 时 间 以 


Ricrosoft SOL Server Danaceseatl Stead| 
文件 到 ) ”编辑 到 ) 视图 查询 @) 项目 EE) 调 寺 0) 
工具 中 窗口 WD 社区 C) 帮助 吕 


卫 新建 查询 W | 昼 洁 久久 | 驴 | 苞 回 马 加 局 


时 地] ost | 执行 Q) 》 有 总 
SQLQueryl. s... ator (52))# vx 
SELECT CHARINDEX (bn apcaeig ]; 

司 结果 | 消息 | 


日 (| TDmm-eszTWaninistrate ., |naster |00:00:00 | 1 行 
就 绪 行 3 列 ! Chl 


图 8.12 ”CHARINDEX() 函 数 的 使 用 


BA 
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及 与 时 间 相 关 的 信息 。 通 常 ， 在 添加 系统 时 间 时 直接 使 用 日 期 时 间 函 数 来 添加 。 常 用 的 日 


期 和 时 间 函 数 如 表 8-3 所 示 。 


表 8-3 日 期 时 间 函 数 


说 明 


GetDate() 


获取 用 户 系统 的 当前 日 期 和 时 间 


Day(date) 


获取 用 户 指定 日 期 date 的 日 数 


Month(date) 


获取 用 户 指定 日 期 date 的 月 数 


Year(date) 


获取 用 户 指定 日 期 date 的 年 数 


DatePart(datepart,date) 


获取 日 期 值 date 中 datepart 指定 的 部 分 值 。datepart 可 以 


是 year、day、week 等 


DateAdd(datepart,num,date) 


在 指定 的 日 期 date 中 添加 或 减少 指定 num 的 值 


DateDiff( datepart,begindate,enddate) 


计算 begindate 和 enddate 两 个 日 期 之 间 的 时 间 间 隔 


实际 上 ， 在 SQL Server 数据 库 中 也 不 止 上 面 列 出 来 的 函数 ， 还 有 其 他 一 些 日 期 函数 ， 
有 兴趣 的 读者 可 以 查看 SQL Server 数据 库 中 的 帮助 文档 查看 相关 的 函数 。 从 表 8-3 中 可 以 
看 出 , 日 期 时 间 函 数 的 使 用 方法 是 很 简单 的 。 下面 就 通过 示例 4 来 演示 如 何 使 用 日 期 函数 。 


【示例 4】 


使 用 日 期 时 间 函 数 完成 如 下 操作 。 
(1) 获取 当前 的 系统 时 间 。 


(2) 获取 当前 系统 时 间 中 的 年 份 。 

(3) 在 当前 时 间 的 基础 上 ， 添 加 10 天 。 

(4) 获取 当前 时 间 到 2013 年 1 月 1 日 的 时 间 间 隔 。 
根据 题目 要 求 ， 只 要 在 表 8-3 中 选择 适合 的 函数 即 可 完成 所 有 操作 。 具 体操 作 如 下 。 
(1) 使 用 GetDate() 函 数 获取 当前 的 系统 时 间 ， 语 句 如 下 : 


SELECT GetDate (); 


运行 结果 如 图 8.13 所 示 。 


(2) 使 用 Year(date) 函 数 来 获取 当前 时 间 的 年 份 ， 语 句 如 下 : 


SELECT Year (GetDate ()); 


运行 结果 如 图 8.14 所 示 。 


文件 下 ) ”编辑 下 ) ”视图 必 ) 查询 伯 ) 
项 目 E) 调试 @) 工具 GD) 窗口 思 ) 


社区 人 C) 帮助 0 


和 2 | 六 | 古今 名 | 训 


a 8 | naster 


| 日 select Gert: 


EE RE 


27\Adninistrato. 


master |00:00:00 |1 行 


就 绪 


图 8.13 ”GetDate() 函 数 的 使 用 


a 


TE 
文件 EE) ”编辑 下 ) ”视图 上 查 淘 外 ) 
项 目 吧 )， 调 坛 @) 工具 外 窗口 员 
社区 加 ) 帮助 0 

及 新 3 查询 WD | 记 | 也 也 可 | 包 蝇 
本 9 三 | naster 区 己 
SQLQueryl. s- - - ator (52))* vx 
| 日 selecc YearlGetDate()); 


2TWaninistrate .. |master |00:00:00 | 1 行 
就 绪 4 Ch 才 


图 8.14 ”Year() 函 数 的 使 用 
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从 图 8.14 的 结果 可 以 看 出 ， 获 取 的 当前 系统 时 间 的 年 份 是 2012 年 。 在 使 用 Year0 函 
数 时 , 不仅 可 以 在 函数 中 使 用 GetDate() 函 数 来 获取 当前 的 系统 时 间 ， 还 可 以 直接 使 用 给 出 
具体 日 期 的 形式 来 表示 ， 比 如 “2012-12-25”。 

(3) 使 用 DateAdd(dateparbnum,date) 函 数 可 以 在 当前 日 期 的 基础 上 加 上 10 天 ， 语 句 
如 下 : 


SELECT DateAdd (day,10,GetDate ()); 


运行 效果 如 图 8.15 所 示 。 
(4) 使 用 DateDiff(datepart,begindate,enddate) 函 数 计算 时 间 间 隔 ， 语 句 如 下 : 


SELECT DateDiff (day,GetDate (),'2013-1-1'); 


运行 效果 如 图 8.16 所 示 。 


icrosoft SQL Server Nanagenent Stadio 
文件 到 ) 编辑 于) 视图 WW) 查询 @) 项 目 @) 调试 @)， 工具 @) 
窗口 和 社区 C) 帮助 0 

也 利于 查询 中 | 站 | 名 访 咏 | 局 | 访 回 当 | 加 : 

By 2 | wester "| 执行) 有 v 8 


文件 到) 编辑 EE) ”视图 上 ) 查询 @@) 项 目 @) 调试 @) 
工具 G) 窗口 岂 社区 人 C) 帮助 00) 


六 旭光 如 | 出 | 轧 也 加 | 记 | 区 吕 忆 加 
RY 2 | waster “| ?执行 OP 加 w 关 


30Uaeryl s... ator (52))* ~ X 
| 日 SELECT DateAdd (day,10,GetDate ()); 


| 
器 结果 | 消息 | 
201301.04 221229780 


ln2008 〔 .，| WIDW-9527\Adninistrato. .，| master |00:00:00 | 1 行 
就 绪 行 3 列 ! Chl 丈 


图 8.15 DateAdd() 函 数 的 使 用 图 8.16 DateDiff) 函 数 的 使 用 
通过 示例 4， 相 信 读 者 已 经 对 日 期 时 间 的 使 用 方法 有 所 了 解 了 。 为 了 能 够 让 读者 对 日 
期 时 间 有 更 深入 的 了 解 ， 请 读者 根据 上 面 的 示例 练习 使 用 DateAdd() 和 DateDiff() 函 数 ， 为 
其 更 换 不 同 的 DatePart 参数 ， 看 看 执行 效果 是 否 满足 您 的 心意 呢 ? 


8.1.4 其 他 函数 


除了 上 面 介绍 的 3 种 类 型 的 函数 外 , 在 SQL Server 中 还 有 一 些 其 他 的 函数 ， 比 如 : 类 
型 转换 函数 、 获 取 系 统 主 要 参数 的 函数 等 。 下 面 就 简单 介绍 一 些 常用 的 函数 。 


1 类 型 转换 函数 


在 SQL Server 中 的 类 型 转换 函数 ,主要 有 两 个 :一 个 是 CONVERTO 函 数 , 一 个 是 CAST() 
函数 。 

(1) CONVERT() 函 数 

CONVERT0) 函 数 主要 用 于 不 同 数据 类 型 之 间 数 据 的 转换 ， 比 如 : 数值 型 转换 成 字符 串 
型 、 字 符 串 类 型 转换 成 日 期 类 型 、 日 期 类 型 转换 成 字符 串 类 型 等 。CONVERT(O 函 数 的 基本 
语法 形式 如 下 所 示 。 


CONVERT ( data type [ ( length ) ] ,expression [ ，style ] ) 


其 中 


站 SELECT DateDiff (day, 
| 
器 结果 | 2 消息 | 
居 芭 至 
NisERvER2008 (.. | WIDW-9527\Adninistrato... |master | 00:00:00 | 1 行 
就 绪 行 8 列 1 ch 1 > 


CD 
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length: 指定 数据 类 型 的 长 度 。 如 果 不 指定 数据 类 


DOODO 


style: 将 数据 转换 后 的 格式 。 由 于 在 实际 转换 中 
因 
的 格式 读者 可 以 参考 SQL Server 的 帮助 文档 ， 这 是 


了 


data_ type: 要 转换 的 数据 类 型 。 比 如 : varchar、float 和 datetime 等 。 


型 的 长 度 ， 则 默认 的 长 度 是 30。 


expression: 表示 被 转换 的 数据 。 可 以 是 任意 数据 类 型 的 数据 。 


经 常会 


指定 日 期 时 间 的 格式 ， 


此 将 日 期 时 间 的 style 格式 给 读者 列 出 ， 供 读者 参考 ， 如 表 8-4 所 示 。 其 他 类 型 


表 8-4 日 期 时 间 型 的 style 值 


有 就 不 一 一 列 出 了 。 


站 带 立 E4 立 的 

的 se 信 Gy) | :9 人 Oyy 区 和 
- 0 或 100 时 mon dd yyyy hh:miAM (或 PM) 
1 101 美国 mm/dd/yyyy 
4 102 ANSI yy.mm.dd 
3 103 dd/mm/yyyy 
4 104 dd.mm.yy 
5 105 dd-mm-yy 
6 106® dd mon yy 
107® mon dd, yy 
8 108 hh:mi:ss 
S 9 或 109 mon dd yyyy hh:mi:ss:mmmAM (或 PM) 
10 110 mm-dd-yy 
11 111 yy/mm/dd 
12 112 ISO . ee I- 
- 13 或 113 dd mon yyyy hh:mi:ss:mmm(24h) 
14 114 |- | hh:mi:ss:mmm(24h) 
- 20 或 120 yyyy-mm-dd hh:mi:ss(24h) 
本 21 或 121 ODBC 规范 〈 带 毫秒 ) | yyyy-mm-dd hh:mi:ss.mmm(24h) 
126 ISO 8601 yyyy-mm-ddThh:mi:ss.mmm (无 空格 ) 
- 127 带 时 区 民 的 ISO 8601 yyyy-mm-ddThh:mi:ssmmmZ (无 空格 ) 
- 130 可 历 dd mon yyyy hh:mi:ss:mmmAM 
- 131 可 历 dd/mm/yy hh:mi:ss:mmmAM 


外 说 明 : 在 实际 应 用 中 ,通常 会 将 年 份 写成 4 位 来 显示 , 也 就 是 使 用 带 世 纪 的 方式 . 因为 ， 


用 两 位 来 表示 年 份 会 造成 一 些 误 读 现象 ， 比 如 : 用 
数据 库 中 会 默认 是 1950 年 ， 想 代表 2050 年 就 不 外 


(2) CAST() 函 数 


CAST() 函 数 与 CONVERT() 函 数 的 作用 是 一 样 的 , 但 是 


一 些 。CASTO 函 数 的 语法 形式 如 下 : 


CAST (expression AS data type [(length)]) 


a 


50 代表 年 份 ， 在 SQL Server 
能 够 直接 写 50 代表 年 份 了 。 


CASTO 函 数 的 语法 形式 更 简单 


26” 


文件 四 ”编辑 EF) 视图 WW) 查询 @) 项 目 E) 调试 @)， 工具 CD) 
窗口 岂 ”社区 C) 帮助 0 窗口 包 社区 C) 帮助 0 
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其 中 : 

口 expression: 表示 被 转换 的 数据 。 可 以 是 任意 数据 类 型 的 数据 。 

口 data type: 要 转换 的 数据 类 型 。 比 如 : varchar、float 和 datetime 等 。 

口 length: 指定 数据 类 型 的 长 度 。 如 果 不 指定 数据 类 型 的 长 度 ， 则 默认 的 长 度 是 30。 
下 面 就 通过 示例 5 来 演示 如 何 使 用 CONVERT() 和 CAST(O 来 转换 数据 类 型 。 

【示例 S】 按照 如 下 要 求 对 数据 类 型 进行 转换 。 

(1) 分 别 使 用 CONVERT(O) 函 数 和 CAST() 函 数 将 当前 日 期 转换 成 字符 串 类 型 。 

(2) 使 用 CASTO 函 数 将 字符 串 1.23 转换 成 数值 类 型 ， 并 保留 一 位 小 数 。 

根据 题目 要 求 ， 按 照 CONVERT0O 函 数 和 CASTO 函 数 的 语法 形式 来 完成 题目 。 

(1) 使 用 CONVERT() 函 数 完成 当前 日 期 转换 成 字符 串 类 型 ， 语 句 如 下 : 

SELECT CONVERT (varchar (20), GetDate (), 111); 

执行 效果 如 图 8.17 所 示 。 

从 图 8.17 的 效果 可 以 看 出 ， 使 用 了 111 的 日 期 格式 ， 转 换 的 字符 串 就 成 了 “2012/12/ 
要 

使 用 CAST() 函 数 将 当前 日 期 转换 成 字符 串 类 型 ， 语 句 如 下 : 


SELECT CAST (GetDate () RS varchar (25)); 


执行 效果 如 图 8.18 所 示 。 


icrosoft SQL Server Wanageneat SA EE =|DIx| 
二 加 区 文件 如、 编辑 全 视图 查询 @) 项 目 @) 调 二 0) 工具 


卫 新建 查询 思 由 | 也 也 下 包 | 区 四 要 过 囊 


和 20 | 访 | 久 十 是 | 记 | 区 加 名 | 风电 pm rr 
昌 master 加 Bd 


ay Wa | naster "| 执行 WW) mv 3 


SELECT CAST (GetDate () AS varchar (25)); 


[aaa | 


NLSERVER2008 〔 .，| WIDYW-9527\Adninistrato. .. msster |00:00:01 | 1 行 


(| NIDW-9527\Adninistrato. .| master |00:00:00 |1 行 


就 绪 4 列 48 Ch 48 行 3 列 1 Ch 1 pA 
图 8.17 使 用 CONVERT(O) 函 数 将 日 期 型 图 8.18 ”使 用 CASTO 函 数 将 日 期 型 转换 
转换 成 字符 串 型 成 字符 串 型 


从 图 8.18 所 示 的 结果 可 以 看 出 , CAST() 函 数 将 日 期 型 转换 成 字符 串 型 的 格式 。 这 个 格 


式 通常 是 不 能 够 指定 的 。 


(2) 使 用 CASTO 函 数 将 字符 串 型 数据 转换 成 数值 型 ， 语 句 如 下 : 


SELECT CAST ('1.23' AS decimal (3,1)); 


执行 效果 如 图 8.19 所 示 。 
这 里 , 读者 也 可 以 自己 尝试 将 本 例 使 用 CONVERTO 函 数 进行 转换 , 并 对 比 转换 后 的 效果 。 


2. 获取 系统 参数 的 常用 函数 
所 谓 系统 参数 ， 是 指 SQL Server 数据 库 所 在 计算 机 的 一 些 信息 以 及 数据 库 的 信息 。 比 
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Er EE 查询 得 ) 项目 @ 调试 四 ) 
工具 四 窗口 WD 社区 加 帮助 四 


ET 


a HR | aster ~| ?执行 GC) 》 则 问 
SQLQuery1- s- vx 
日 SELECT ' aS decimal(3,1)); py 
器 结果 | 2 消息 | 
玩 列 名 ] 


bos C.. | WIDW-9527\Adninistrato. .. |master |00:00:00 |1 行 
就 绪 第 1! 行 ”第 1 列 


图 8.19 ”使 用 CASTO 函 数 将 字符 串 类 型 数据 转换 成 数值 型 数据 
如 : 计算 机 名 称 、 数 据 库 户 名 以 及 应 用 程序 名 称 等 。 常 用 的 函数 如 表 8-5 所 示 。 
表 8-5 ”获取 系统 参数 函数 


序号 函数 形式 说 明 

1 HOST _ NAME 获取 数据 库 所 在 的 计算 机 名 称 

2 HOST ID() 获取 数据 库 所 在 计算 机 的 标识 号 

3 DB _ NAME([database id]) 获取 数据 库 名 称 ，database id 表示 数据 库 的 ID 号 

4 DB ID([database name']) 获取 数据 库 的 标识 号 ，database_name 表示 数据 库 的 名 称 

5 APP NAME() 获取 当前 会 话 的 应 用 程序 名 称 

6 USER NAME([id]) 获取 数据 库 用 户 的 名 称 ，id 是 与 数据 库 用 户 关联 的 标识 号 
7 USER_ID([user]) 获取 数据 库 用 户 的 标识 号 ，user 是 数据 库 用 户 名 

8 SUER SNAME([user sid]) 获取 数据 库 的 登录 名 ，user_sid 是 数据 库 用 户 的 ID 号 

上 面 的 表格 中 列 出 的 都 是 比较 常用 的 获取 系统 参数 的 函数 ， 如 果 需 要 学 习 其 他 的 一 些 


函数 可 以 参考 SQL Server 的 帮助 文档 中 的 内 容 。 此 外 ， 以 上 的 8 个 函数 ， 请 读者 自行 在 
SQL Server 的 环境 中 熟悉 它们 的 用 法 ， 以 便 在 今后 的 学 习 和 工作 中 更 好 地 应 用 它们 。 


8.2” 自 定义 函数 


在 上 一 节 中 ， 已 经 将 SQL Server 中 提供 的 一 些 常用 函数 做 了 详尽 的 介绍 了 。 读 者 可 以 
想 一 想 ， 如 果 我 们 要 实现 一 些 功能 ， 在 SQL Server 提供 的 函数 列表 中 找 不 到 怎么 办 呢 ? 正 
所 谓 ， 自 己 动手 丰衣足食 嘛 ， 也 就 是 说 ， 要 自己 来 定义 函数 嗪 。 在 本 节 中 ， 就 带领 读者 学 
习 如 何 来 自 定 义 函数 。 
8.2.1 创建 自 定义 函数 的 语 ; 

在 实际 应 用 中 ， 如 果 没 有 可 用 的 系统 函数 供 选择 ， 通 常 可 以 自己 来 创建 函数 。 那 么 如 
何 来 创建 自 定义 函数 呢 ? 还 是 要 遵循 系统 函数 的 形式 来 创建 的 。 自 定义 函数 主要 分 为 两 种 ， 


一 种 是 标量 函数 ， 即 通过 计算 得 到 一 个 具体 的 数值 ; 一 种 是 表 值 函数 ， 即 通过 函数 返回 数 
据 表 中 的 查询 结果 。 常 用 的 自 定义 函数 是 标量 函数 。 创 建 自 定义 函数 的 语法 如 下 所 示 。 
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(1) 标量 值 函数 的 语法 结构 


CREATE FUNCTION function name (@parameter name parameter data type...) 
RETURNS return data type 
[AS] 
BEGIN 
function body 
RETURN scalar expression 
END 


其 中 : 

function_name 项 : 用 户 定义 函数 的 名 称 。 

@parameter_name 项 : 用 户 定义 函数 的 参数 ， 函 数 最 多 可 以 有 1024 个 参数 。 
parameter_data_type 项 : 参数 的 数据 类 型 。 

return_data_type 项 : 标量 用 户 定义 函数 的 返回 值 类 型 。 

function_body 项 : 指定 一 系列 定义 函数 值 的 工 SQL 语句 。 

scalar_expression 项 : 指定 标量 函数 返回 的 标量 值 。 

(2) 内 联 表 值 函数 的 语法 结构 

CREATE FUNCTION function name (@parameter _ name parameter data type...) 
RETURNS TABLE 


[ RS ] 
RETURN [ ( ] select stmt [ ) ] 


DOOOODDO 


其 
口 function_name 项 : 用 户 定义 函数 的 名 称 。 

口 @parameter_name 项: 用 户 定义 函数 的 参数 ， 函 数 最 多 可 以 有 1024 个 参数 。 
口 parameter_data_type 项 : 参数 的 数据 类 型 。 

口 TABLE 项 :指定 表 值 函数 的 返回 值 为 表 。 

口 select_stmt 项 ， 定 义 内 联 表 值 函 数 的 返回 值 的 单个 SELECT 语句 。 


8.2.2” 先 建 一 个 没有 参数 的 标量 函数 


无 参数 的 函数 也 是 经 常会 用 到 的 ， 比 如 : 获取 当前 系统 的 时 间 。 按 照 标量 函数 的 创建 
语法 ， 在 标量 函数 中 既 可 以 带 参 数 也 可 以 不 带 参数 ， 下 面 就 以 示例 6 为 例 学 习 如 何 创 建 一 
个 没有 参数 的 标量 函数 。 本 章 中 所 有 的 函数 都 将 创建 在 chapter8 数据 库 中 。 

【示例 6】 创建 标量 函数 ， 计 算 当 前 系统 年 份 被 2 整除 后 的 余数 。 

根据 题目 要 求 , 直接 使 用 系统 函数 是 不 方便 的 。 在 创建 标量 函数 时 , 返回 值 是 余数 值 ， 
创建 语句 如 下 : 

create function funl() 

returns INT 

本 

return CRAST (Year (GetDate ()) RS INT) %2 

end 

执行 上 面 的 语句 ， 就 可 以 在 chapter8 数据 库 中 创建 一 个 名 为 funl 的 函数 了 。 那 么 ， 有 
了 函数 该 如 何 调用 这 个 函数 呢 ? 这 就 与 直接 调用 系统 函数 类 似 了 ， 但 是 也 略 有 不 同 。 在 调 
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用 自 定 义 函 数 时 ， 需 要 在 该 函数 前 面 加 上 dbo。 下 面 就 来 调用 新 创建 的 函数 fun1， 具 体 语 
名 如 下 : 


SELECT dbo.funl(); 


执行 效果 如 图 8.20 所 示 。 
从 图 8.20 所 示 的 结果 可 以 看 出 ， 返 回 值 是 0。 也 就 是 说 2012%2 等 于 0。 


8.2.3 ”再 建 一 个 带 参数 的 标量 函数 


通过 上 面 的 示例 6， 相 信 读 者 对 创建 函数 以 及 如 何 调用 函数 也 有 所 了 解 了 ， 下 面 继续 
学 习 如 何 创建 带 参数 的 标量 函数 。 带 参数 的 标量 函数 不 论 是 在 创建 还 是 调用 时 ， 都 与 无 参 
函数 的 使 用 有 一 些 区 别 。 请 读者 认真 体会 示例 7， 看 看 它们 究竟 有 什么 不 一 样 呢 ? 

【示例 7】 创建 标量 函数 ， 传 入 商品 价格 作为 参数 ， 并 将 传 入 的 价格 打 八 折 。 

根据 题目 要 求 ， 将 商品 价格 打 八 折 ， 也 就 是 将 商品 价格 乘 以 0.8 即 可 。 具 体 的 语句 
如 下 : 

create function fun2 (eprice decimal (6,2)) 

returns decimal (6,2) 

begin 

return @price*0.8 

end 

执行 上 面 的 语句 ， 就 可 以 在 数据 库 chapter8 中 创建 名 为 fun2 的 函数 了 。 在 示例 6 中 学 
习 了 如 何 调用 无 参 的 函数 ， 那 么 ， 带 参数 的 函数 如 何 调用 呢 ? 下 面 就 将 答案 告诉 你 了 

假设 要 打 八 折 的 商品 价格 是 2100 元 ， 那 么 调用 函数 的 语句 如 下 : 


SELECT dbo.fun2(2100); 


执行 上 面 的 语句 ， 效 果 如 图 8.21 所 示 。 


> Microsol ft SQL Server Managemert aa 阿 [=] 太 了 | 


编辑 (E) ”视图 VW。 查询 (Q) 
项 目 (6) 调试 D) 工具 (D 窗口 
社区 (CO) 夫 助 (H) 

也 新 尘 查 询 (V) | 四 中 也 本 此 记号 遇 
9? a | chaptere 


SQLQuery1.sql --istrator (52))* SR 
日 SELECT dbo.fun1(); 


-9527\Administrato,.，| chapter8 | 00:00:00 | 1 行 


匹配 : begin 行列 1ch: 
图 8.20 ”调用 函数 fun1 


文件 (E) ”编辑 (E) ”视图 (V) 查询 (Q) 

项 目 (P) 调式 D) 工具 (D 窗口 (WwW) 

社区 (QO 帮助 (H) 

和 2 新 津 可 济 V 出 | 也 也 四 让 | 所 下 
By 13 | chaptere 入 


SQLQuery1.s-ator (52))* vx 
昌 SELECT dbo.fun2 (2100); 一 
| = 


器 结果 | 2 消息 | 
项 列 名 ) 
1658000 | 


-9527\VAdministrato.,，| chapter8 | 00:00:00 | 1 行 


匹配 : beon 行列 Ich: 


图 8.21 调用 函数 fun2 


通过 图 8.21 的 调用 结果 可 以 看 出 ,在 调用 带 参数 的 函数 时 必须 要 为 其 传递 参数 ， 并且 
参数 的 个 数 以 及 数据 类 型 要 与 函数 定义 时 的 一 致 。 


8.2.4 创建 表 值 函数 
表 值 函数 与 标量 函数 一 样 ， 既 可 以 带 参数 也 可 以 不 带 参 数 。 使 用 表 值 函数 ， 通 常 是 为 


。184 。 


了 完 
函数 
户 信 


习 
uj 
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成 根据 某 一 个 条 件 ， 查 询 出 相应 的 查询 结果 。 下 面 就 使 用 示例 8 来 演示 如 何 使 用 表 值 
。 在 使 用 表 值 函数 之 前 ， 先 要 在 数据 库 chapter8 中 准备 一 张 数据 表 ， 这 里 创建 一 张 用 
息 表 ， 表 结构 如 表 8-6 所 示 。 
表 8-6 用 户 信息 表 (userinfo) 

数据 类 型 中 文 释义 


INT 用 户 编号 


wm ||- 


张 表 
如 下 


name VARCHAR(20) 用 户 名 
password VARCHAR(20) 密码 
根据 表 8-6 的 结构 创建 数据 表 后 ， 录 入 表 8-7 所 示 的 数据 。 
表 8-7 用 户 信息 表 数 据 


用 户 编号 (id) 密码 (password ) 
! 123456 
2 123456 


好 了 ， 准 备 工作 已 经 完成 了 。 下 面 就 进入 示例 8 的 学 习 了 。 

【示例 8】 创建 表 值 函数 ， 根 据 输入 的 用 户 编号 ， 查 询 出 该 用 户 的 用 户 名 以 及 密码 。 
根据 题目 要 求 ， 该 表 值 函数 含有 一 个 INT 类 型 的 参数 ， 创 建 语句 如 下 : 

create function fun3(@id int) 

returns table 

ee select name,password from userinfo where id=@id; 

执行 上 面 的 语句 ， 即 可 在 数据 库 chapter8 中 创建 函数 fun3。 由 于 表 值 函数 的 结果 是 一 
， 因 此 , 在 调用 的 时 候 也 是 直接 用 SELECT 语句 查询 就 可 以 了 。 调用 fun3 函数 的 语句 


SELECT * FROM dbo.fun3 (1); 


执行 上 面 的 语句 ， 效 果 如 图 8.22 所 示 。 


可 Microsal ft SQL Server gemeni LIofxj 
文件 E) 编辑 (@) 视图 ( 凡 ， 查 询 (Q) 项目 P) 

调试 D) 工具 (D 窗口 (WwW) 社区 (QO 帮助 (H) 

卫 新 建 查询 | 六 | 启 渤 名 | 启 | 蕊 回忆 
人 | chapter8 = 
ne | 
日 SELECT * FRON dbo.fun3(1); | 


[Jame | password | | 


DW-9527\Administrato,.. | chapter8 | 00:00:00 | 1 行 
行列 1chl 去 


图 8.22 ”调用 函数 fun3 
从 图 8.22 所 示 的 结果 可 以 看 出 ， 向 fun3 函数 中 传递 1 作为 参数 ， 得 到 数据 表 中 id 为 


“Se 
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1 的 数据 。 
8.2.5 修改 自 定 义 函 数 


自 定义 函数 也 是 可 以 修改 的 ， 修 改 语句 与 创建 语句 很 相似 ， 只 是 将 创建 自 定义 函数 语 
法 中 的 CREATE 语句 换 成 ALTRE 语句 就 可 以 了 。 具 体 的 语法 读者 直接 参考 创建 自 定义 函 
数 的 语句 ， 这 里 就 不 袭 述 了 。 下 面 用 示例 9 讲解 如 何 修改 自 定义 函数 。 

【示例 9】 修改 自 定义 函数 fun3， 改 成 根据 编号 查询 出 用 户 名 。 

根据 题目 要 求 ， 修 改 fun3 的 语句 如 下 : 

alter function fun3(@id int) 

returns table 


as 
return select name from userinfo where id=Q@id; 


执行 上 面 的 语句 , chapter8 中 的 函数 fun3 就 已 经 被 修改 了 。 调 用 fun3 函数 的 语句 如 1 


select * from dbo.fun3(1); 


执行 效果 如 图 8.23 所 示 。 ee 
相信 读者 通过 示例 9， 已 经 了 解 如 何 修改 自 定义 函 开口 IZAD mo tE@ mw 


数 了 。 请 读者 练习 修改 函数 fanl 和 fun2。 一 
8.2.6 去除 自 定义 函数 


如 果 自 定义 函数 不 再 需要 了 ， 为 了 节省 数据 库 占用 。 | 于 谨 委 | 
的 空间 ， 要 及 时 地 删除 没有 用 的 自 定义 函数 。 无 论 是 标 。 howssznwannarao.， [we noo0 |1 生 
量 函 数 还 是 表 值 函数 ， 删 除 的 语句 都 是 一 样 的 ， 具体 语 。 一 一 一 一 
句 如 下 : 图 8.23 ”调用 修改 后 的 fun3 函数 


了 


DROP FUNTION dbo.fun name; 

这 里 ， 值 得 注意 的 是 删除 函数 前 ， 要 先 打 开 函 数 所 在 的 数据 库 。 

下 面 就 使 用 示例 10 来 演示 如 何 删除 自 定义 函数 。 

【示例 10】 删除 函数 fun1。 

根据 题目 要 求 ， 删 除 函 数 的 语句 如 下 : 

DROP FUNCTION dbo.funl; 

执行 上 面 的 语句 ， 函 数 funl 就 从 数据 库 chapter8 中 删除 了 。 
8.2.7 ”在 企业 管理 器 中 也 能 管理 自 定义 函数 

在 前 面 的 小 节 中 ,都 是 使 用 SQL 语句 来 创建 和 管理 自 定义 函数 的 。 实际 上 , 使 用 企业 
管理 器 也 能 完成 同样 的 功能 。 也 就 是 说 ， 如 果 一 时 想 不 起 来 创建 自 定义 函数 的 语法 ， 就 可 


以 借助 企业 管理 器 中 的 提示 来 帮助 你 唆 ! 下 面 就 以 创建 、 修 改 和 删除 fnnl 函数 为 例 ， 讲 解 
如 何在 企业 管理 器 中 操作 自 定义 函数 。 
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1. 在 企业 管理 器 中 创建 fun1 函数 


funl 函数 是 计算 当前 系统 年 份 被 2 整除 后 的 余数 。 在 企业 管理 器 中 创建 funl 函数 , 只 
需要 使 用 鼠标 展开 chapter8 数据 库 节 点 ， 并 展开 其 下 的 “可 
编程 性 ”节点 ， 在 其 节点 下 就 可 以 看 到 “函数 ”的 节点 。 如 sr 
图 8.24 所 示 。 田 国 视图 

从 图 8.24 所 示 的 界面 中 ,可 以 看 出 函数 包括 了 表 值 函数 、 a se 
标量 值 函数 、 聚 合 函数 以 及 系统 函数 。 这 里 ，funl 属于 标量 = 已 B 
值 函数 ， 因 此 创建 在 “标量 值 函 数 ”的 节点 下 。 Rn 


示 “标量 值 函数 ”节点 ， 在 田 加 系统 函数 
在 图 8.24 所 示 的 界面 中 ， 右 击 “ 标 量 值 函数 ”节点 ， 在 2 


弹出 的 右键 菜单 中 选择 “新 建 标量 值 函数 "选项, 出 现 图 8.25 i 
所 示 界 面 。 


udia 二 加权 
项 目 (E) ”调式 (D) 工具 (D 窗口 (Ww) 社区 (QO 帮助 ) 
辽 新 建 查询 由 章 | 售 六 加 局 | 臣 回转 | 总 5 
-| 1 扫 5W 》 瑞 Y 路 邓 国 | ” 码 | 多 贺 友 1 三 全 | 池 字 | 仿 曲 
x 5 2.sql -inistrator (53)) | ”SQLQuery1.5q ~,,.istrator (52))* | -x 
SET ANSI_NULLS ON 司 


日 SET QUOTED_IDENTIFIER ON 
田 加 系统 数据 库 pa 


sg 
日 国 chaptere -- Author: <Author, ,Name> 

田 国 数据 库 关系 图 eate date: <Create Date, ,> 

田 留 表 scription: <Descrtipci 

田 镶 视 目 ne 


n_Name, sysname, FunctionName 


-- Add the parameters for the function here 
QParaml, sysname, Bpl> <Data Type_For_Paraml, , int 


RETURNS <Function Data_ Type, ,int: 
-- Declare the return variable here 
DECLARE shResultVar, sysname, BResult> <Function Data Type, ,int: 


-- hdd the T-SQL statements to compute the return Value here 


田 加 计划 指南 SELECT sRhesultyar, aysname, RResult> ~ “BParami, sysnae, Rpl> 
田园 Service Broker 
田 国 存储 ~- Return che result of the funccion 


RETURN <BResultVar, sysname, BReault 


上 
划 | 才 已 连接 。(UD) | WIDW-9527\MS5QLSERVER2008 (..，| wWIDW-9527WAdministrato.，| chaptere | 00:00:00 | 0 行 
就 绪 行 22 列 2 cz Ins 


图 8.25 ”新 建 标量 值 函数 界面 


在 图 8.25 所 示 的 界面 中 ， 读 者 就 可 以 发 现 创建 表 量 值 函数 的 语法 框架 已 经 显示 出 来 
只 需要 添加 具体 的 内 容 就 可 以 了 。 下 面 就 将 fun1 函数 的 功能 填 入 图 8.25 所 示 的 界面 
中 。 效 果 如 图 8.26 所 示 。 

确认 图 8.26 填 入 的 信息 后 ， 保 存 函 数 信息 即 可 。 至 此 ，funl 函数 就 创建 成 功 了 。 


2. 在 企业 管理 器 中 修改 fun1 函数 
修改 函数 信息 ， 相 对 于 创建 函数 还 是 要 简单 一 些 的 。 在 企业 管理 器 中 chapter8 数据 库 


. 187， 


本 
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二 
文件 (E) ”编辑 (E) ”视图 (WW 项目 (e) 调试 D) 工具 (D 窗口 (wW) 社区 (QO 帮助 (由 


EE 
TCR 下 下 TH Ta TI 


- Use the Specify Values for Template Parameters 
-- command (Ctrl-Shift-N) to fill in the parameter 
一 values below. 

-- This block of comments will not be included in 
-— the definition of the function. 


SET ANSI_NULLS ON 
Go 
SET QUOTED_IDENTIFIER ON 
Go 
日 -- = 
Author: <Author, ,Name> 
Create date: <Creace Date, ,> 
Description: <Description, ,> 
章 CREATE FUNCTION fun1() 
RETURNS int 
as 
BEGIN 


-- Return che result of the function 
RETURN CAST|IYear (GetDate()) AS INT) 


图 8.26 创建 funl 函数 的 界面 


里 ， 单 击 “ 可 编程 性 ”|“ 标 量 值 函 数 ” 选 项 ， 并 在 标量 值 函数 列表 中 右 击 “funl1” 函 数 ， 
在 弹出 的 右键 菜单 中 选择 “修改 ”选项 ， 即 可 出 现 图 8.27 所 示 的 界面 。 


osoft SQLS nt Studio =|DIx| 
文件 E) ”编辑 (E) ”视图 (VW) 查询 (9) 项 目 P) 调试 D) 工具 (D 窗口 (Ww) 社区 (QO 帮助 (H) 

;也 .新建 查询 | 丹 | 六 池 串 |D | 芒 回 加 5 

3 | chapter ~| 9 执行 O hh 昌 w ?3 地 
EF 国 | “5QLQuery3.sql --istrator (55))*| SQL Query2.sq .nistrator (53)) | zx 


r Manag 


USE [chapter8] ~ 
B WIDW-9527\MS5QLSERVER2008 (天 Go 
日 为 数据 库 /x+ssfs Object: UserDefinedFunction [dbo] . [funl] Script I 
加 系统 数据 库 SET ANSI_NULLS ON 
为 数据 库 快 昭 Go 
日 Bchapters SET QUOTED_IDENTIFIER ON 
田 国 数据 库 关系 图 Go 
田 国 表 BALTER function [dbo] .[funl] () 
田 国 视图 returns INT 
田 国 同义词 as 
日 加 可 编程 性 begin 
各 存储 过 程 return CASTIYear (GetDate()) AS INT) s2 
日 国 函数 end 


田 国 表 值 函数 
日 国 标量 值 函 数 
加 只 don 
田 嘱 dbo.funz 

田 疗 来 全 函数 
田 国 系统 函数 +, 
和 | 于 已 | wipw-9sz7WM5sQLsERVER2008 (.… | WIDW-9527\Administrato.,，| chapter8 | 00:00:00 | 0 行 
行 16 列 1 hl ms 用 


图 8.27 ”修改 funl 函数 界面 
在 图 8.27 所 示 的 界面 中 ， 对 funl 函数 修改 后 ， 保 存 即 可 完成 对 funl 函数 的 修改 。 


"188。 
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3. 删除 fun1 函数 

删除 函数 的 操作 相对 于 添加 和 修改 就 更 简单 了 .在 企业 管理 器 中 的 chapter8 数据 库 里 ， 
单 击 “ 可 编程 性 ”|“ 标 量 值 函数 ”选项 ， 右 击 “fun1” 函 数 ， 并 在 弹出 的 右键 菜单 中 选择 
“删除 ”选项 ， 弹 出 图 8.28 所 示 界 面 。 


EE 
服务 器 
YTDW-9527\MSSQLSERYER2006 


连接 
WIDN-9527\Adninistr ator 
于 查看 连接 悍 性 


进度 
就 绪 


Cw | ww | 
图 8.28 删除 funl 函数 界面 


在 图 8.28 所 示 的 界面 中 ， 单 击 “确定 ”按钮 ， 即 可 将 函数 funl 删除 。 
从 上 面 的 3 种 操作 中 ， 读 者 已 经 看 出 企业 管理 器 给 你 带 来 的 便利 了 吧 。 虽 然 在 企业 管 
理 器 中 操作 函数 很 方便 ,但 是 读者 也 不 要 忽视 对 SQL 语句 的 学 习 哦 ! 


全 说 明 : 实际 上 ， 在 企业 管理 器 中 ， 不 仅 可 以 对 函数 进行 创建 、 修 改 以 及 删除 的 操作 ， 还 
可 以 对 函数 进行 重 命名 、 脚 本 编辑 等 操作 。 另 外 ， 还 要 提醒 读者 的 是 在 创建 不 同 
类 型 的 函数 时 ， 一 定 要 创建 在 相应 的 函数 文件 中 ! 


8.3 本 章 小 结 


在 本 章 中 主要 学 习 了 SQL Server 中 的 系统 函数 以 及 如 何 创建 和 使 用 自 定义 函数 。 在 系 
统 函 数 部 分 主要 给 读者 介绍 了 常用 的 数学 函数 、 字 符 串 函数 、 日 期 时 间 函 数 以 及 其 他 的 函 
数 。 在 自 定义 函数 部 分 主要 给 读者 分 别 讲解 了 如 何 使 用 SQL 语句 和 在 企业 管理 中 创建 和 管 
理 自 定义 函数 。 函 数 作为 数据 库 的 主要 组 成 部 分 ， 如 果 能 够 在 实际 的 工作 中 运用 自如 ， 一 
定 会 起 到 事半功倍 的 作用 的 ! 请 读者 一 定 要 多 看 、 多 练 、 多 想 ， 在 实际 工作 中 创建 出 更 多 
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的 给 你 带 来 便利 的 函数 。 
8.4 本 章 习 题 


一 、 填 空 题 


1， 系统 函数 主要 包括 

2.， 自 定义 函数 主要 包括 

3. 创建 自 定义 函数 的 语句 是 

二 、 选 择 题 

1. 下 列 关于 自 定 义 函 数 的 描述 正确 的 是 
A. 自 定义 函数 可 以 重 名 
B.， 自 定义 函数 必须 有 参数 
C. 自 定义 函数 可 以 有 0 到 多 个 参数 


D. 以 上 都 不 对 
2. 取 绝 对 值 的 函数 是 8 

A. ABS() B. EXP() C. ABSS(O) D. 以 上 都 不 对 
3. 取 字 符 串 长 度 的 函数 是 

A. count() B. LENO C. LONG() D. 以 上 都 不 对 
三 、 问 答题 


1. 系统 函数 中 字符 串 函数 都 有 哪些 ? 
2. 表 值 函数 与 标量 函数 的 区 别 是 什么 ? 
3， 如何 删除 自 定义 函数 ? 


这 3 裔 纶 遍 放 位 用 交 了 
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视图 从 字面 上 的 意思 理解 可 以 成 “可 以 看 见 的 图 ”， 也 就 是 说 是 图 而 不 是 表 哦 。 视 图 
既然 不 是 表 ， 那 它 在 数据 库 中 扮演 着 什么 角色 呢 ? 视图 与 表 的 操作 非常 相似 ， 它 实际 上 是 
由 查询 1 张 或 多 张 表 的 查询 语句 所 组 成 的 对 象 。 在 本 章 中 就 将 学 习 如 何 使 用 视图 。 

本 章 的 主要 知识 点 如 下 : 

口 视图 的 概念 
如 何 创 建 视图 
如 何 更 新 视图 
如 何 删 除 视图 
使 用 DML 语句 操作 视图 


9.1 了 和 解 视图 


视图 也 经 常 被 很 多 人 称 为 “虚拟 的 表 ”， 所 谓 虚拟 ， 就 不 是 真实 存在 的 东西 。 这 就 好 
像 是 网 络 游戏 中 的 虚拟 货币 一 样 ， 这 些 货币 并 不 是 真实 的 人 民 币 ， 但 是 却 可 以 在 网 络 游戏 
中 购买 东西 。 但 是 ， 这 些 网 络 中 的 虚拟 货币 却 可 以 通过 人 民 币 来 购买 ， 并 且 还 可 以 将 网 络 
游戏 中 的 货币 转换 成 真实 的 人 民 币 。 

视图 中 的 数据 全 部 都 来 源 于 数据 库 中 的 1 张 或 多 张 数据 表 。 通 常情 况 下 ， 将 组 成 视图 
中 的 数据 表 称 为 源 表 或 基 表 。 那 么 ， 既 然 视 图 是 一 张 虚拟 的 表 ， 为 什么 数据 库 中 还 有 这 样 
的 对 象 呢 ?下 面 就 告诉 读者 视图 究 竞 能 给 你 带 来 哪些 好 处 吧 。 

(1) 降低 SQL 语句 的 复杂 程度 

在 SQL Server 中 ， 如 果 要 查询 的 数据 来 自 于 多 张 表 ， 就 需要 用 多 表 的 联合 查询 了 。 如 
果 在 SQL 语句 中 过 多 地 使 用 多 表 联 合 查询 ， 就 会 使 SQL 语句 看 起 来 复杂 一 些 了 。 那 么 ， 
使 用 视图 ， 将 经 常 需要 多 表 连 接 查 询 的 语句 保存 到 视图 中 ， 以 后 再 查询 同样 的 信息 ， 就 能 
够 直接 通过 视图 查询 了 。 举 个 例子 来 说 说 吧 。 如 图 9-1 所 示 ， 表 1、 表 2 是 学 生成 绩 表 和 
科目 信息 表 。 表 3 是 从 表 1、 表 2 查询 的 数据 构成 的 。 

从 图 9.1 所 示 的 效果 来 看 ， 要 查询 的 是 学 生 的 成 绩 信 息 ， 由 于 学 生成 绩 表 中 的 科目 编 
号 列 是 与 科目 信息 表 的 序号 列 对 应 的 。 因 此 ， 在 查询 时 需要 使 用 多 表 连 接 查 询 。 具 体 的 查 
询 语句 如 下 : 

SELECT a. 编 号 , a. 姓名 ,a. 学 号 ,pb. 科 目 名 称 , a. 成 绩 


FROM 学 生成 绩 表 a, 科目 信息 表 b 
WHERE a. 科 目 编号 =b. 序 号 ; 


口 
口 
口 
口 


表 1 学 生成 绩 表 
柄 避 9 | 姓名 + 学 可 EE 
1? | 王 医 ” DO0le 1 S00 
20 | 齐 云 ” 00027 25 805 局 
:| 吉本 襄 Oo 表 3 学 生 入 息 视 国 中 的 帮 据 , 
5 E43 0005?” 39 EE [号 号 ”| 姓名” 字号 ” 科目 名 称 ” 成语] 
ip ES 00017 计算 机 基础 | 80% | 
表 2 科目 导 息 表 " > EE EE 003 GC 语言 5 Eo 
下 一 - 35 | 地政 00035 计算 机 革 各 5 | 555 | 
序号 ”| 科目 名 称 " -下 4 可 00045 CC 语言 55 | 
1 计算 机 基础 ” | 55 | 于 鸯 ” 0005” 数据 库 * 795 | 


35 CC 语言” 


和 | | 


图 9.1 视图 的 形成 


如 果 将 上 面 的 命令 存放 到 学 生 信息 视图 中 ， 就 可 以 直接 查询 出 表 3 的 结果 了 。 查 询 视 
图 和 查询 表 的 语句 类 似 ， 语 句 如 下 : 

SELECT * FROM 学 生 信 息 视 图 ; 

从 上 面 的 语句 可 以 看 出 ， 通 过 视图 已 经 将 查询 语句 从 3 行 变 成 了 1 行 了 。 确 实 简化 
了 吧 。 

(2) 提高 数据 库 的 安全 性 

一 个 小 小 的 视图 ， 居 然 还 能 够 提高 数据 库 的 安全 性 。 那 就 看 看 它 是 如 何 给 数据 库 提 高 
安全 性 的 吧 。 所 谓 数据 库 的 安全 性 ,也 是 数据 表 的 安全 性 。 如 果 直 接 在 数据 表 中 查询 数据 ， 
在 查询 语句 中 就 会 涉及 到 数据 表 的 名 称 和 列 名 ， 这 样 就 给 数据 表 的 安全 带 来 了 隐患 。 如 果 
将 数据 表 的 查询 命令 放 到 视图 中 存放 ， 那 么 ， 使 用 视图 查询 数据 时 就 可 以 避免 数据 表 名 称 
泄露 了 。 因 此 ， 使 用 视图 是 可 以 提高 数据 库 的 安全 性 的 。 

(3) 便于 数据 共享 

数据 共享 也 可 以 理解 成 是 数据 对 于 每 个 人 来 说 都 公用 的 ， 谁 都 可 以 拿 来 使 用 。 当 需要 
根据 不 同 条 件 查 询 一 张 数 据 表 时 ， 数 据 库存 取 速 度 就 会 下 降 。 而 将 数据 表 的 不 同 查 询 命令 
放 到 多 个 视图 中 存放 时 ， 每 次 查询 都 只 查询 视图 ， 这 样 就 在 数据 共享 的 基础 上 提高 了 查询 
速度 。 

任何 一 个 事务 都 会 有 优点 也 有 缺点 ， 视 图 也 不 例外 ， 它 也 是 有 缺点 的 。 比 如 : 在 创建 
视图 时 不 能 够 使 用 GROUP BY、HAVING 等 子 语句 ， 在 视图 中 存放 的 命令 包含 多 张 表 时 ， 
不 能 够 直接 更 新 视图 信息 。 


9.2 创建 视图 


在 上 一 节 中 ， 已 经 清楚 了 视图 的 基本 概念 以 及 视图 给 我 们 带 来 的 一 些 好 处 。 创 建 视图 
是 使 用 视图 的 第 一 个 步骤 。 视 图 既 可 以 由 一 张 表 组 成 也 可 以 由 多 张 表 组 成 。 在 本 节 中 就 将 
带领 你 完成 创建 视图 的 操作 。 

9.2.1 创建 视图 的 语法 


创建 视图 的 语法 与 创建 表 的 语法 一 样 ， 都 是 使 用 CREATE 语句 来 创建 的 。 在 创建 视图 
时 ， 只 能 用 到 SELECT 语句 。 具 体 的 语法 如 下 : 


Co 
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CREATE VIEW view name 
AS select_statement 
[WITH CHECK OPTION] 
[ENCRYPTION] 
其 中 : 
口 view_name: 视图 的 名 称 。 在 一 个 数据 库 中 视图 的 名 称 也 是 不 能 重复 的 。 通常 视图 
的 名 称 都 是 以 “V_” 开 头 的 。 
口 AS: 指定 视图 要 执行 的 操作 。 
口 select_statement: 用 于 定义 视图 的 查询 语句 。 该 语句 可 以 使 用 多 个 表 和 其 他 视图 。 
在 视图 中 使 用 的 数据 表 被 称 为 基 表 或 者 源 表 。 
口 CHECK OPTION: 强制 执行 对 视图 的 数据 修改 语句 , 都 必须 符合 在 select_statement 
中 设置 的 条 件 。 该 选项 是 可 选 的 。 
口 ENCRYPTION: 对 创建 视图 的 语句 加 密 。 该 选项 是 可 选 的 。 
9.2.2 源 自 一 张 表 的 视图 
根据 一 张 表 创建 视图 通常 都 是 选择 一 张 表 中 的 儿 个 经 常 需要 查询 的 字段 。 下 面 就 对 在 
本 章 中 要 使 用 的 数据 表 进 行 说 明 。 在 本 章 中 共 使 用 两 张 数 据 表 ， 并 把 这 两 张 数 据 表 创 建 在 


数据 库 chapter9 中 。 这 两 张 数 据 表 分 别 是 学 生成 绩 信息 表 以 及 科目 信息 表 。 具 体 表 结 构 如 
表 9-1、 表 9-2 所 示 。 


表 9-1 学 生成 绩 信息 表 (studentinfo) 


编号 列 名 数据 类 型 中 文 释义 
1 id INT 编号 (主键 ) 
2 studentid INT 学 号 
3 name VARCHAR(20) 姓名 
4 major VARCHAR(20) 专业 
5 subjectid INT 科目 编号 
6 score DECIMAL(5,2) 成 绩 
7 remark VARCHAR(200) 备注 

表 9-2 科目 信息 表 (subjectinfo) 

编号 列 名 数据 类 型 中 文 释义 
1 id INT 科目 编号 
pe subject VARCHAR(20) 科目 名 称 


根据 上 述 的 表 结构 ， 创 建 数 据 表 的 语句 如 下 : 


USE chapter9; 
CREATE TABLE studentinfo 
( 
id int identity(1,1) PRIMARY KEY, 
studentid int, 
name varchar (20) ， 
major varchar (20), 
subjectid int, 
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score decimal (5,2), 
remark varchar (200) 


CREATE TABLE subjectinfo 

( 

Id int identity (1, 1) PRIMARY KEY, 
subject varchar (20), 

i 


通过 上 面 的 语句 ,就 可 以 将 学 生成 绩 信息 表 和 科目 信息 表 创 建 在 chapter9 数据 库 中 了 。 
创建 好 数据 表 后 ， 将 表 9-3、 表 9-4 中 的 数据 分 别 添加 到 数据 表 中 。 


表 9-3 ”学 生成 绩 信息 表 中 的 数据 


编号 学 号 
201201 
201210 


1 

2 王 明 85 

3 201215 
4 

5 


20025 | 末末 | S| 3 
zs | 汪 十 | | 4 | » | 


科目 编号 名 称 
1 线性 代数 
2 毛泽东 思想 概论 运筹 学 
3 


有 了 前 面 的 数据 准备 ， 下 面 就 要 开始 学 习 如 何 创建 视图 了 。 
【示例 1】 创建 视图 v_studentinfo， 用 于 查看 学 生 的 学 号 以 及 姓名 、 所 在 专业 。 
根据 题目 要 求 ， 要 查询 的 信息 都 在 学 生成 绩 信息 表 中 ， 创 建 视图 的 语句 如 下 : 


CREATE VIEW v_studentinfo 


RS 
SELECT studentid as 学 号 ,name as 姓名 ,major as 所 在 专业 FROM studentinfo 
执行 上 面 的 语句 ， 就 可 以 在 数据 库 chapter9 中 创建 视图 了 。 执 行 效果 如 图 9.2 所 示 。 


了 microseft SQL Server Nanagenent Studio =Iolx| 

文件 四 ”编辑 有) 视图 WD 查询 @) 项 目 E) 调式 0) 工具 CD) 窗口 WD 社区 帮助 0 

辽 新建 查询 | 六 | 渤 迎 名 | 让 区 回忆 | 台 s 

Wy a | chapter9 |? 执行 D 有 v 史 可 且 ) 对 | 的 贿 友 | 三 全 名 
~ 


SQLQuery3. s...ator (51))#| 
唱 CREATE VIEW Vv studentinfto 


a5 
[Ser atudentid as 学 号 ,name as 姓名 ,major as 所 在 专业 FRONM studentinfo 


园 查询 已 成 功 执 … | WIDW-9527\ISSQLSERYER2008 〔 .| WID-9527\Adninistrato. .. | chapter9 | 00:00:00 |0 行 
就 绪 行 9 列 1 ch1 Is 


图 9.2 创建 视图 v_studentinfo 


“ 现 9 
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9.2.3 源 自 多 张 表 的 视图 


所 谓 源 自 多 张 表 的 视图 ， 也 就 是 说 视图 中 的 数据 是 从 多 张 数据 表 查 询 出 来 的 。 体 现 视 
图 中 的 数据 来 源 于 多 张 表 ， 主 要 就 是 更 改 SQL 语句 。 这 回 读者 应 该 有 信心 完成 了 吧 。 下 面 
就 用 示例 2 来 演示 如 何 建立 源 自 多 张 表 的 视图 。 

【示例 2】 创建 视图 v_studentinfo2， 用 于 查询 学 生 的 姓名 、 专 业 、 科 目 名 称 以 及 成 绩 。 

根据 题目 要 求 ， 要 从 学 生成 绩 信息 表 和 科目 信息 表 中 查询 数据 。 有 具体 语句 如 下 : 


CREATE VIEW v studentinfo2 
RS 
SELECT studentinfo.name as 姓名 ,studentinfo.major as 所 在 专业 ， 


subjectinfo.subject as 科目 名 称 , studentinfo.score as 成 绩 
FROM studentinfo, subjectinfo 
where studentinfo.subjectid=subjectinfo.Id; 


执行 上 面 语句 ， 即 可 创建 视图 v_studentinfo2。 执 行 效果 如 图 9.3 所 示 。 


a Microsoft SQL Server Management Studio =|DIxl 
文件 E) ”编辑 (E) 视图 ( 几 ” 查 询 (9) 项目 (E) 调试 D) 工具 (D 窗口 (Ww) 社区 (GO 帮助 () 

辽 新建 查询 由 | 也 也 下 | 记 | 记 加 本 | 名 

9 Qa | chapter -| ?#5 》 mv 汶 电 回 | 半 码 | 鸡 圈 的) 三 全 | 幸 字 | 各 局 


“SQLQuery1.sql -istrator (52))*| ER 
日 CREATE VIEW v_studentinfo2 


AS 

SELECT atudentinfo.name as 姓名 ,atudentinfo.major as 所 在 专业 ,，subjectinfo.subject as ncn 
studentinfo.score as 成 绩 

了 RON stu tinfo,subjectinfo 

Where studentinfo,.subjectid=subjectinfo.Id; 


1 » 


后 消息 | 

命令 已 成 功 完成 。 
加 小 
团 查 询 已 成 功 执行 。 | WIDW-9527\MSSQLSERVER2008 (,.，| WIDW-9527\Administrato..，| chapter9 | 00:00:00 | 0 行 
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图 93 创建 视图 v_studentinfo2 


从 图 9.3 所 示 的 结果 可 以 看 出 ， 源 自 多 张 表 的 视图 只 是 SQL 语句 改变 了 。 


外 说 明 :查询 视图 中 的 数据 与 查询 数据 表 中 的 数据 是 一 样 的 ， 都 是 使 用 SELECT 语句 来 
查询 。 


9.3 更 新 视图 


无 论 是 买 东西 还 是 完成 工作 计划 ， 都 免不了 需要 修改 一 些 东西 。 比 如 : 在 商场 买 了 一 
条 裤子 ， 如 果 裤 子 过 长 还 是 会 去 修改 一 下 裤 脚 的 。 视 图 也 不 例外 ， 当 创建 了 一 个 视图 后 ， 
觉得 有 些 地 方 需要 改进 还 是 可 以 修改 的 ， 而 不 需要 重新 创建 视图 。 


9.3.1 更 新 视图 的 语法 
在 SQL Server 中 ， 更 新 视图 的 语句 与 创建 视图 的 语句 非常 类 似 ， 具 体 的 语句 如 下 : 
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ALTER VIEW view name 
AS select statement 

[WITH CHECK OPTION] 

[ENCRYPTION] 


从 上 面 的 语法 中 , 读者 可 以 看 出 除了 将 创建 视图 的 CREATE 关键 字 换 成 ALTER 之 外 ， 
其 他 的 语法 都 是 一 样 的 。 读 者 如 果 对 上 面 的 参数 还 有 不 清楚 的 地 方 ， 可 以 参考 创建 视图 的 
语法 解释 。 


9.3.2 ”视图 很 容易 改 


看 了 上 一 小 节 的 更 新 视图 的 语法 ， 相 信 读 者 已 经 迫不及待 地 想 尝 试 一 下 如 何 使 用 这 个 
语法 了 。 那 么 ， 下 面 就 以 示例 3 来 演示 如 何 修改 视图 。 

【示例 3】 修改 在 示例 1 中 创建 的 视图 v_studentinfo， 将 其 改 成 只 显示 学 生 的 姓名 和 
专业 。 

根据 题目 要 求 ， 修 改 视图 的 语句 如 下 : 

ALTER VIEW v_studentinfo 

AS 

SELECT name as 姓名 ,major as 所 在 专业 FROM studentinfo; 

执行 上 面 的 语句 ， 即 可 完成 对 视图 v_studentinfo 的 修改 。 效 果 如 图 9.4 所 示 。 


,Microsoft SQL Server Management Studio =|Djxl 
文件 E) ”编辑 (E) 视图 查询 (Q) 项 目 P) 调试 D) 工具 (D 窗 D(W) 社区 (O 
帮助 (H) 


有 | 让 | 古本 怨 | 访 | 瑟 回 沁 | 加 局 
对 好 | chapter -| 1》 v 鸣 名 国 | 下 鸣 久 
5QLQuery2.sql -istrator (53))*| ~x 
日 ALTER VIEW Vv studentinfo 

有 SS 

SELECT name as 姓名 ,major as 所 在 专业 FRONM studentinfo; 国 


轧 消息 | 
命令 已 成 功 完成。 = 
下 » 


WIDW-9527\MSSQLSERVER2008 (,.，| WIDW-9527WAdministrato,,，| chapter9 | 00:00:00 | 0 行 
行 4 


就 绪 列 ! Chl 曙 


图 9.4 修改 v_studentinfo 界面 


在 图 9.4 所 示 的 界面 中 ， 就 可 以 看 出 修改 视图 的 语句 很 容易 记 吧 ， 只 需要 将 创建 视图 
语句 中 的 CREATE 改 成 ALTER 就 可 以 嗪 ! 
9.3.3 ”给 视图 换个 名 字 


给 视图 换 名 字 ， 就 是 我 们 通常 所 说 的 视图 重 命名 。 重 命名 视图 要 使 用 系统 存储 过 程 
sp_rename 来 实现 。 下 面 就 使 用 一 个 示例 来 演示 如 何 给 视图 换 名 字 。 

【示例 4】 将 视图 v_studentinfo 的 名 字 改 成 v_studentinfo1。 

根据 题目 要 求 ， 给 视图 重 命名 的 语句 如 下 : 


sp_rename 'v_studentinfo', 'v_studentinfol'; 
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执行 上 面 的 语句 ， 即 可 完成 对 视图 的 重 命名 。 效 果 如 图 9.5 所 示 。 


寺 、 Microsol ft SQL Server Management Studio =Io|x| 
文件 E) 编辑 (E) ” 视 苞 Y) 查 渔 (9) 项 目 P) 调试 D) 工具 D 窗 DW 社区 (O 
帮助 (H) 
卫 新 建 查 W 出 | 也 也 一 包扎 加 本 台所 
Wy a | chapter9 -| ?5W bv 于 时 国 | 站 鸡 久 
SQLQuery2.sql -istrator (53))*| vx 
日 sp_rename 'v_studentinfo', 'v_ studentinfol'; 


已 消息 | 
注意 : 更 改 对 象 名 的 任 一 部 分 都 可 能 会 破坏 脚本 和 存储 过 程 。 


ul 
| WIDW-9527\MS5QLSERVER2008 (| WIDW-9527\Adrministrato,.，| chapter9 | 00:00:00 | 0 行 
就 结 行 4 列 1 Chl 肥 


图 9.5 视图 的 重 命名 
从 图 9.5 所 示 的 结果 可 以 看 出 ， 在 对 视图 进行 重 命 名 后 会 给 使 用 该 视图 的 程序 造成 一 
定 的 影响 。 因 此 ， 在 给 视图 重 命名 前 ， 先 要 知道 是 否 有 一 些 其 他 数据 库 对 象 使 用 该 视图 名 
称 。 在 确保 不 会 对 其 他 对 和 象 造 成 影响 后 ， 再 对 视图 名 称 进行 修改 ! 
全 说 明 : 视图 除了 使 用 系统 存储 过 程 对 其 更 名 ， 也 可 以 使 用 系统 存储 过 程 sp_refreshview 
对 其 进行 刷新 操作 。 刷 新 视图 的 目的 就 在 于 更 新 视图 的 查询 结果 。 


9.4 删除 视图 


不 论 任何 数据 库 对 象 都 会 占用 数据 库 的 存储 空间 的 ， 因 此 视图 也 不 例外 。 当 视图 不 再 
使 用 时 ， 要 及 时 删除 数据 库 中 多 余 的 视图 。 当 然 ， 如 果 不 确 定 该 视图 以 后 是 否 使 用 时 ， 要 
对 视图 先 备 份 然后 再 删除 哦 ! 

9.4.1 删除 视图 的 语法 

删除 视图 的 语法 很 简单 ， 但 是 在 删除 视图 前 一 定 要 确认 视图 是 否 不 再 使 用 了 ， 和 否则 删 
除 后 就 不 能 恢复 了 。 删 除 视图 仍然 使 用 DROP 关键 字 来 完成 。 删 除 视图 的 语法 如 下 : 

DROP VIEW [ schema name . ] view name [ ...,n] [;] 

其 中 : 

口 schema_name 项 : 指 该 视图 所 属 架 构 的 名 称 。 


口 view_name 项 : 指 要 删除 的 视图 的 名 称 。 
这 里 ，schema_name 是 可 以 省 略 的 。 


9.4.2 ”删除 不 用 的 视图 


下 面 就 应 用 删除 视图 的 语法 ， 使 用 示例 5 来 演示 如 何 删除 数据 库 中 不 用 的 视图 。 
【示例 S】 删除 视图 v_studentinfo1。 
根据 题目 要 求 ， 删 除 视图 的 语句 如 下 : 


-lhe 


DROP VIEW v_studentinfol; 


执行 上 面 的 语句 ， 指 定 的 视图 即 可 从 数据 库 chapter9 中 删除 了 。 效 果 如 图 9.6 所 示 。 


[noror sat Server Maraoer nfe EA 
文件 (E) ”编辑 (E) ”视力 (Y) ”查询 (9) 项 上 (PE) 
调试 D) 工具 (D 窗口 (W) 社区 (QO 帮助 (中 


了 as 局 | 古 钙 名 || 瑟 


BY 3 | chapter9 | 
SQLQuery2.sql -..istrator (53))* Xx 
日 DROP VIEW Vv _ studentinfol; a 
时 | "TW 
已 消息 | 
命令 已 成 功 完成 。 
可 
IDW-9527\Administrato..，| chapter9 | 00:00:00 | 0 行 
就 绕 行列 1Ch1 jy 


图 9.6 删除 视图 v_studentinfol 


9.5 使 用 DML 语句 操作 视图 


DML 语句 是 数据 操纵 语言 ， 是 对 数据 表 进 行 操作 的 。 那 么 ，DML 语言 能 够 在 视图 中 
使 用 吗 ? 答案 是 肯定 的 。 但 是 ， 也 并 不 是 说 所 有 的 视图 都 能 够 使 用 DML 语言 来 操作 视图 
中 的 数据 。 在 本 节 中 就 将 带领 读者 学 习 如 何 使 用 DML 语言 来 操作 视图 。 


9.5.1 使 用 INSERT 语句 向 视图 中 添加 数据 


视图 是 一 张 虚拟 的 数据 表 ， 实 际 上 在 视图 中 是 不 保存 数据 的 。 那 么 ， 怎 么 向 视图 中 添 
加 数据 呢 ? 读者 不 要 忘记 了 ， 视 图 中 的 数据 是 来 源 于 基 表 的 ， 也 就 是 向 基 表 中 添加 数据 。 
但 是 ， 这 样 一 来 并 不 是 所 有 的 视图 都 能 够 使 用 INSERT 语句 来 添加 数据 的 ， 只 有 当 视 图 中 
的 基 表 唯一 并 且 在 视图 中 使 用 的 字段 是 直接 使 用 基 表 的 字段 ， 而 不 是 通过 其 他 形式 派生 的 
字段 ， 比 如 : 使 用 聚合 函数 或 其 他 的 函数 。 

在 视图 中 使 用 INSERT 语句 ， 与 操作 表 时 使 用 INSERT 语句 的 语法 是 一 样 的 ， 因 此 ， 
如 果 读 者 忘记 了 INSERT 语句 的 语法 ， 可 以 参考 本 书 第 3 章 的 相关 内 容 。 

在 前 面 的 示例 2 中 ， 创 建 了 视图 v_studentinfo2， 并 且 该 视图 是 由 两 张 基 表 组 成 的 。 下 
面 就 使 用 示例 6 来 演示 INSERT 语句 在 v_studentinfo2 上 的 应 用 。 读 者 想 想 这 种 INSERT 操 
作 会 成 功 吗 ? 

【示例 6】 使 用 INSERT 语句 向 视图 v_studentinfo2 中 添加 一 条 数据 。 

根据 题目 要 求 ， 添 加 数据 的 语句 如 下 : 

INSERT INTO V_studentinfo2 

VALUES (' 章 小 ',' 计 算 机 ',' 数 据 库 '，, 88); 

执行 上 面 的 语句 ， 效 果 如 图 9.7 所 示 。 

从 图 9.7 所 示 的 结果 可 以 看 出 ， 由 多 张 基 表 组 成 的 视图 ， 是 无 法 使 用 INSERT 语句 对 
其 更 新 的 。 


Ss 


第 3 篇 数据 库 使 用 进 阶 


站 
文件 日 ” 妨 辑 (E) 视图 四。 查询 (Q) 项目) 调式 D) 工具 (D 窗口 (WD 

社区 (QO 帮助 

巡 新 时 可) | 六 | 半 这 名 | 届 | 蕊 四 对 | 加 5 

By Wa | chapter? -| 1 执 5W 》 和 v 哮 晤 国名 
SQLQuery2.sql -istrator (53))*| vx 


日 INSERT INTO v_studentinfo2 


varozs(' 章 小 '… 计 算 机 ' 数据 库 ' .88) ; 


局 消息 | 
俏 息 4405， 级 别 16， 状 态 1, 第 1 行 < 
视图 或 函数 'v_studentintoz， 个 可 更 新 ， 因 为 修改 会 影响 多 个 基 表 。 局 
yl 者 
IDW-9527\MS5QLSERVER2008 (… | WIDW-9527\Administrato... | chapter9 | 00:00:00 | 0 行 
就 绪 行 6 列 1 chl 4 


图 9.7 向 由 多 张 基 表 组 成 的 视图 添加 数据 


既然 已 经 知道 了 由 多 张 基 表 组 成 的 视图 是 不 能 够 使 用 INSERT 语句 对 其 添加 数据 的 ， 
下 面 再 通过 示例 7 来 演示 如 何 使 用 INSERT 语句 向 单 基 表 组 成 的 视图 添加 数据 。 

【示例 7】 创建 视图 v_studentinfo3， 查 询 科 目 信息 表 的 科目 名 称 ， 并 使 用 INSERT 语 
句 向 该 视图 中 添加 一 条 数据 。 

根据 题目 要 求 ， 创 建 视图 的 语句 如 下 : 

CREATE VIEW v_studentinfo3 


AS 
SELECT subject FROM subjectinfo; 


» 


执行 上 面 的 语句 ， 即 可 在 数据 库 chapter9 中 创建 视图 v_studentinfo3。 
有 了 视图 v_studentinfo3， 下 面 使 用 INSERT 语句 向 其 添加 一 条 数据 ， 语 名 如下: 


INSERT INTO V_studentinfo3 
VALUES (' 电 子 技术 ') ， 


执行 上 面 的 语句 ， 效 果 如 图 9.8 所 示 。 Ey 
从 图 9.8 所 示 的 结果 可 以 看 出 ， 使 用 INSERT 文件 护 加 ( 视图 查询 (Q) ”项 目 (B) 调式 D) 


语句 是 可 以 直接 向 符合 条 件 的 视图 中 添加 数据 的 。 和 


BY 2 | dhapter9 ”| 执行 2) 》 


9.5.2 使 用 UPDATE 语句 更 新 视图 中 的 。 | ow = 
数据 2 


VALUES (' 电 子 技术 ') ; 

UPDATE 语句 与 INSERT 语句 在 视图 中 的 使 用 i 
方法 是 类 似 的 ， 修 改 视图 的 语句 与 修改 数据 表 的 语 ” 贺 型 
名 也 是 一 样 的 。 下 面 就 用 示例 8 来 演示 如 何 使 用 。 ee ome [oo es 
PAB 全 让 于 宙 人 轩 由 国 娄 的 图 9.8 向 单 基 表 组 成 的 视图 中 添加 数据 

【示例 8】 创建 视图 v_studentinfo4， 从 
studentinfo 表 查 询 学 生 的 学 号 (studentid) 和 姓名 (name)， 并 将 学 号 是 201201 的 学 生 姓 
名 更 改 成 “吴琼 ”。 

根据 题目 要 求 ， 创 建 视图 v_studentinfo4 的 语句 如 下 : 


CREATE VIEW v_studentinfo4 
AS 


已 消息 | 
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SELECT studentid, name FROM studentinfo; 


执行 上 面 的 语句 ， 视 图 v_studentinfo4 在 数据 库 chapter9 中 就 创建 好 了 。 
有 了 视图 Vv_studentinfo4， 使 用 UPDATE 语句 更 新 视图 的 语句 如 下 : 


UPDATE v_studentinfo4 
SET name=' 吴 琼 ' WHERE studentid=201201; 


执行 上 面 的 语句 ， 效 果 如 图 9.9 所 示 。 
从 图 9.9 所 示 的 结果 可 以 看 出 ， 使 用 UPDATE 语句 是 可 以 完成 视图 数据 的 更 新 的 。 


9.5.3 使 用 DELETE 语句 删除 视图 中 的 数据 


通过 前 面 的 INSERT 和 UPDATE 语句 的 讲解 ， 相 信 读 者 对 使 用 DELETE 语句 删除 视 
图 中 数据 的 操作 已 经 能 够 猜测 出 来 了 。 没 错 ， 使 用 DELETE 语句 也 只 能 对 单 基 表 的 视图 进 
行 操 作 。 下 面 就 通过 示例 9 来 演示 如 何 使 用 DELETE 语句 删除 视图 中 的 数据 。 

【示例 9】 删除 视图 v_studentinfo4 中 ， 名 字 为 “吴琼 ”的 学 生 信息 。 

根据 题目 要 求 ， 删 除 语句 如 下 : 

DELETE FROM Vv studentinfo4 

WHERE name=' 吴琼 '; 


执行 上 面 的 语句 ， 效 果 如 图 9.10 所 示 。 
从 图 9.10 所 示 的 结果 中 ， 可 以 看 出 删除 单 基 表 组 成 的 视图 中 的 数据 是 可 以 通过 
DELETE 语句 来 完成 的 。 


Microsoft SQL Server Management Studig 二 = 上 着 
文 作 (编辑 E) ”视图 查询 (Oo) 项目 (6) ”调式 (0) 工具 

窗口 (W) ”社区 (CO 和 助 t 册 

了 新建 查 询 U | 让 | 售 偿 芒 | 访 | 苞 回 对 加 5s 

; MY 3 | chapter9 "| mv 了 允 回 镶 


“SQLQuery2.sql -..istrator (53))* vx 
日 UPDATE v_studentinfo4 
LSET name=! 吴 琼 ， WHERE studentid=201201; 


[Es Microsort sat Server Managerient ta Te Ed 
文件 (E) ”编辑 (E) ”视图 (v) 查询 (Q) 项目 (p) 
调试 D) 工具 (D 窗口 (WW) 社区 (QO 帮助 (t) 
卫 新 建 查询 | 记 | 也 也 四 | 记 | 世 加 名 
时 ”1 | chapter9 ”| 执行) 区 
SQL Query2.sql -istrator (53))* 


加 消息 | 


41 行 受 影响 ) 
可 


155QLSERVER2008 (,,，| WIDW-9527\Administrato,,，| chapter9 | 00:00:00 | 0 行 
就 绪 行 4 列 1 chl 友 


WIDW-9527\Administrato.,.，| chapter9 | 00:00:00 | 0 行 
就 绪 行 4 列 1 chl 多 


图 9.9 使 用 UPDATE 语句 更 新 视图 v_studentinfo4 图 9.10 使 用 DELETE 语句 删除 视图 中 的 数据 


9.6 使 用 企业 管理 器 操作 视图 


在 本 章 前 面 的 内 容 中 ， 都 是 使 用 SQL 语句 来 操作 视图 的 。 实 际 上 , 在 企业 管理 器 中 同 
样 可 以 完成 对 视图 的 操作 ， 包 括 创 建 视图 、 修 改 视图 以 及 删除 视图 等 。 在 本 节 中 就 将 带领 
读者 学 习 如 何在 企业 管理 器 中 操作 视图 。 
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9.6.1 使 用 企业 管理 器 创建 视图 


在 企业 管理 器 中 创建 视图 ， 就 免 去 了 记 住 SQL 语句 的 麻烦 。 下 面 就 使 用 示例 10 来 演 
示 如 何在 企业 管理 器 中 创建 视图 。 

【示例 10】 创 建 视图 v_studentinfo5, 查询 出 学 生 信息 表 (studentinfo) 中 学 号 (studentid)、 
姓名 (name) 以 及 专业 (major) 信息 。 

在 企业 管理 器 中 创建 视图 ， 需 要 通过 如 下 4 个 步 又。 

(1) 在 企业 管理 器 中 ， 展 开 chapter9 数据 库 ， 右 击 “ 视 图 ”节点 ， 在 弹出 的 右键 菜单 
中 选择 “新 建 视图 ”选项 ， 弹 出 “添加 表 ” 对 话 框 ， 如 图 9.11 所 示 。“ 添 加 表 ” 对 话 框 中 
有 当前 数据 库 中 存在 的 表 、 视 图 、 函 数 和 同义词 对 象 。 

(2) 在 图 9.11 所 示 的 界面 中 ,选择 要 创建 视图 使 用 的 数据 表 studentinfo， 单 击 “ 添 加 ” 
按钮 ， 并 单 击 “ 关 闭 ” 按 钮 关闭 “添加 表 ” 对 话 框框 。 显 示 效 果 如 图 9.12 所 示 。 


Microsoft SQL Server Management Studia =|DIxl| 
文 作品 贺 避 加 视图 项 目 C) 调 碟 D) 查询 设 计 跟 R) 工具 (D 
OW EO WD 

了 W290 | 鹿 | 古 太太 | 所 | 蕊 回 台 | 凤 忆 
回 四 加 自 必 eI 


WIDW-9527\MS... dbo.view_1*| 


FE [到 Hm 
表 


subjectinfo 


就 绪 
图 9.11 “添加 表 ” 对 话 框 图 9.12 ”添加 表 后 的 效果 
外 说 明 : 如 果 要 选择 多 个 表 ， 必 须 在 图 9.11 所 示 的 界面 中 单 击 鼠 标 选择 表 的 同时 ， 按 下 


Ctrl 键 。 


(3) 在 图 9.12 所 示 界 面 中 ， 设 置 或 编写 视图 中 的 查询 语句 。 视 图 中 的 查询 语句 ， 可 以 
直接 在 选中 的 表格 中 选择 ， 也 可 以 直接 编写 SQL 语句 。 这 里 ， 直 接 在 数据 表 中 使 用 鼠标 选 
择 studentid、name 以 及 major 列 。 设 置 后 的 效果 如 图 9.13 所 示 。 

(4) 通过 前 面 3 个 步 又， 视图 已 经 创建 完成 了 。 但 是 ， 最 后 一 步 也 不 能 忽视 ， 就 是 保 
存 视图 。 单 击 工具 栏 中 的 “保存 ”按钮 ， 弹 出 “选择 名 称 ” 对 话 框 ， 如 图 9.14 所 示 。 输 入 
视图 名 称 v_studentinfo5， 单 击 “ 确 定 ”按钮 ， 即 可 完成 创建 视图 的 操作 了 。 


9.6.2 ”使 用 企业 管理 器 修改 视图 
修改 视图 的 界面 与 创建 视图 非常 类 似 , 闲话 少 叙 ,下 面 就 以 示例 11 为 例 与 读者 共同 学 
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了 Microsoft SQL Server Management Studig -olx] 


文件 编 贺 视图) 项目) 调 坛 D) 查询 设计 器 (8) 工具 
全 Do 社区 (下 助 内 
;也 新 建 林山 | 出 | 仿 辽 怠 记 | 区 加 忆 | 双 后 


上 回回 加 Ee1 


| WIDW-9527\MS... dbo.View_1* 三 X 


表 输出 | 排序 类 型 之 
studentinfo 区 
studentinfo 区 
Studentinfo 区 
[3 


SELECT studentid, name major 
FROM 。 dbo,studentnfo 


图 9.13 设置 视图 中 的 SQL 语句 


习 如 何在 企业 管理 器 中 修改 视图 。 


图 9.14 “选择 名 称 ” 对 话 框 


【示例 11】 使 用 企业 管理 器 修改 视图 v_studentinfo5， 使 其 只 查询 出 学 生成 绩 信 息 表 
(Cstudentinfo) 中 的 学 生 姓 名 (name) 和 专业 (major)。 

在 企业 管理 器 中 创建 视图 ， 需 要 通过 如 下 3 个 步骤 。 

(1) 在 企业 管理 器 中 ， 选 择 视 图 所 在 的 数据 库 chapter9， 展 开 “ 视 图 ”节点 ， 查 找到 
要 修改 的 视图 v_studentinfo5。 右 击 该 “视图 ”节点 ， 在 弹出 的 右键 菜单 中 选择 “设计 ” 命 


令 ， 如 图 9.15 所 示 。 


(2) 修改 视图 中 的 语句 。 在 图 9.15 所 示 的 界面 中 ， 从 数据 表 中 去 掉 studentid 的 选项 ， 


如 图 9.16 所 示 。 


SQL Server Management Studio 
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窗口 (WwW) 社区 (GO 和 和 觅 () 

及 和 2 村 池 V) | 六 | 渤 六 芒 | 访 | 芒 回 各 骂 5 
回回 加 四 要 eI 


Bs Microsoft SQL Server Manaoernent Sti 二 6 本 
文件 E) ”编辑 (E) ”视图 (W 项 目 (E) ”调试 D) 查询 设计 器 (R) 工具 (TD) 
窗口 (W) 社区 (C) ”帮助 (H) 
PE 


回 图 回国 ?四 LT 站 


WIDW-9527\MS.._studentinfoS 三 X 
站 


| WIDW-9527\MS.tudentinfos* x 


4 » 人 » 
列 EE ETE" am ma 
studentid studentinfo name studentinfo 回 
name Studentinfo 区 major Studentinfo a 
» maor studentinfo Ea 上 所 
区 多 了 | 


ml | 了 到 


SELECT studentid, name, major 
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图 9.15 v_studentinfo5 视图 的 设计 界面 


已 保存 的 项 


图 9.16 修改 后 的 v_studentinfo5 视图 


SS 
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(3) 单 击 工具 栏 中 的 “保存 ”按钮 ， 即 可 完成 对 v_studentinfo5 视图 的 修改 操作 。 


全 说 明 : 在 企业 管理 器 中 也 可 以 修改 视图 的 名 称 ， 右 击 “ 视 图 ”节点 ， 选 择 “ 重 命名 ” 命 
令 ， 直 接 修改 视图 名 称 即 可 。 但 是 ， 视 图 可 不 能 重 名 的 ! 


9.6.3 ”使 用 企业 管理 器 删除 视图 


删除 视图 应 该 是 对 视图 操作 中 最 简单 的 一 种 操作 了 。 不 需要 再 使 用 视图 设计 器 操作 ， 
直接 选择 视图 就 可 以 删除 了 。 下 面 就 通过 示例 12 来 演示 如 何 删除 视图 。 

【示例 12】 删除 视图 v_studentinfo5 。 

删除 视图 v_studentinfo5， 需 要 通过 以 下 两 个 步骤 完成 。 

(1) 在 企业 管理 器 中 ， 展 开 chapter9 数据 库 ， 选 择 要 删除 的 视图 v_studentinfo5， 右 击 
该 视图 名 称 ， 在 弹出 的 右键 菜单 中 ， 选 择 “删除 ”命令 。 弹 出 如 图 9.17 所 示 的 对 话 框 。 
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图 9.17 删除 视图 提示 对 话 框 
(2) 在 提示 对 话 框 中 ， 单 击 “ 确 定 ” 按 钮 即 可 将 视图 v_studentinfo5 删除 。 


9. 7 本 章 小 结 已 


在 本 章 中 主要 讲解 了 SQL Server 中 视图 的 创建 、 更 新 以 及 删除 操作 ， 并 且 还 讲解 了 使 
j DML 语句 如 何 操作 视图 。 在 使 用 DML 语句 操作 视图 时 ， 不 仅 要 掌握 相关 的 DML 语法 
规则 ， 还 要 掌握 和 辨识 什么 样 的 视图 是 可 以 通过 DML 语句 操作 的 。 此 外 ， 视 图 的 操作 也 

是 可 以 通过 企业 管理 器 直接 单 击 操作 完成 的 , 在 忘记 SQL 语句 时 , 不 要 忘记 这 个 方便 的 工 
具 哦 ! 
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一 、 填 空 题 

1. 创建 视图 的 关键 字 是 。 

2. 视图 中 的 数据 可 以 来 源 于 张 表 。 
3. 给 视图 重 命名 使 用 的 是 。 

二 、 选 择 题 


1. 下 列 关于 视图 的 描述 正确 的 是 
A. 视图 中 的 数据 全 部 来 自 于 数据 库 中 存在 的 表 
B. 使 用 视图 可 以 方便 地 查询 数据 
C. 视图 经 常 被 称 为 “虚拟 的 表 ” 


D. 以 上 都 对 

2. 下 列 修改 视图 的 语句 正确 的 是 
A. renew view_name B. drop view_name 
C. alter view_name D. 以 上 都 不 对 

3. 关于 使 用 DML 语句 操作 视图 的 描述 正确 的 是 8 


A. 不 能 向 视图 中 插入 数据 

B. 可 以 向 任意 视图 插入 数据 

C. 只 能 向 由 一 张 基 表 构成 的 视图 中 插入 数据 
D. 以 上 都 不 对 


、 问 答题 


1. 视图 的 作用 是 什么 ? 
2. 如 何 使 用 企业 管理 器 操作 视图 ? 
3. 如 何 使 用 DML 语句 操作 视图 ? 


四 、 操 作 题 


通过 本 章 的 学 习 ， 完 成 如 下 对 视图 的 操作 。 以 表 9-1 所 示 的 学 生成 绩 信 息 表 
(Cstudentinfo) 为 例 。 

(1) 创建 视图 ， 查 询 studentinfo 表 中 的 学 生 名 称 和 专业 。 

(2) 向 (1) 中 创建 的 视图 添加 一 条 数据 。 

(3) 查询 〈1) 中 创建 的 视图 。 

(4) 删除 (1) 中 创建 的 视图 。 
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索引 这 个 词 听 起 来 有 点 陌生 , 但 是 目录 大 家 应 该 都 听 说 过 吧 。 没 错 ,在 SQL Server 2008 
中 索引 就 与 图 书 上 的 目录 很 相似 ， 目 录 是 为 了 让 读者 更 快 地 查找 所 需 的 内 容 ， 那 么 ， 索 引 
也 就 是 帮助 数据 库 操作 人 员 更 快 地 查找 数据 库 中 的 数据 叶 。 

本 章 的 主要 知识 点 如 下 : 

口 认识 索引 

口 创建 索引 

口 修改 索引 

口 删除 索引 


10.1 认识 索引 


在 本 章 的 开头 已 经 说 过 索引 与 目录 的 作用 是 相似 的 , 那么 , 在 SQL Server 2008 中 索引 
究 疯 有 什么 用 呢 ? 另外 ， 索 引 都 分 为 几 类 呢 ? 下 面 就 请 读者 在 本 节 中 找到 上 面 两 个 问题 的 
答案 。 


10.1.1 索引 的 作用 


索引 在 数据 库 检 索 中 有 着 举足轻重 的 作用 ， 试 想 一 下 如 果 一 本 书 没有 目录 ， 查 找 资料 
时 就 需要 从 第 一 页 开始 翻 ， 多 么 可 怕 啊 。 如 果 在 数据 表 中 没有 索引 ， 也 是 可 以 查找 数据 的 
但 是 会 花费 更 长 的 时 间 。 在 任何 时 候 ， 任 何人 都 想 通过 数据 库 快速 查找 资料 ， 因 此 ， 使 用 
索引 是 很 有 必要 的 。 索 引 是 建立 在 数据 表 中 列 上 的 一 个 数据 库 对 象 ， 在 一 张 数据 表 中 可 以 
给 一 列 或 多 列 设置 索引 。 如 果 在 查询 数据 时 ， 使 用 了 设置 的 索引 列 作为 检索 列 ， 那 么 就 会 
大 大 提高 查询 速度 。 


10.1.2 ”索引 就 这 么 几 类 


在 SQL Server 2008 数据 库 中 , 索引 主要 分 为 聚集 索引 和 非 聚集 索引 两 类 。 在 一 张 数据 
表 中 只 有 一 个 聚集 索引 。 这 就 好 像 人 只 能 有 一 个 身份 证 号 是 一 样 的 。 具 体 的 使 用 方法 如 下 
所 示 。 
口 聚集 索引 : 最 常见 的 聚集 索引 就 是 主键 约束 。 它 根据 数据 行 的 键 值 在 表 或 视图 中 
排序 和 存储 这 些 数据 行 。 
口 非 聚集 索引 : 非 聚 集 索引 在 一 张 表 中 可 以 有 多 个 。 它 包含 非 聚集 索引 键 值 ， 并 且 
每 个 键 值 项 都 有 指向 包含 该 键 值 的 数据 行 的 指针 。 
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10.2 创建 索引 


创建 索引 是 使 用 索引 的 第 一 步 ， 前 面 在 学 习 索 引 类 型 时 已 经 清楚 了 索引 有 非 聚 集 索 引 
和 聚集 索引 两 种 ， 因 此 ， 在 创建 索引 前 也 要 和 弄 清楚 要 创建 的 是 哪 种 类 型 的 索引 。 在 本 节 中 
将 带领 读者 使 用 语句 和 企业 管理 器 来 创建 不 同类 型 的 索引 。 


10.2.1 创建 索引 的 语法 


创建 索引 与 创建 表 一 样 ， 都 是 创建 数据 库 对 象 ， 因 此 ， 仍 然 使 用 CREATE 语句 。 在 创 
建 索引 的 语法 中 就 包括 了 创建 聚集 索引 和 非 聚集 索引 这 两 种 方式 ， 读 者 可 以 根据 需要 自行 
选择 。 具 体 的 语法 如 下 : 


CREATE [ UNIQUE ] [ CLUSTERED | NONCLUSTERED ] INDEX index_name 
ON 
[ database name]. table or view name (column [ RSC | DESC ] [,...n ]) 


其 
口 UNIQUE: 唯一 索引 。 

口 CLUSTERED: 聚集 索引 。 

口 NONCLUSTERED: 非 聚 集 索引 。 

口 index_name: 索引 的 名 称 。 索 引 名 称 在 表 或 视图 中 必须 唯一 。 

口 column: 索引 所 基于 的 一 列 或 多 列 。 由 多 列 组 成 的 索引 被 称 为 组 合 索引 。 
口 [ASCIDESC]: 确定 索引 列 的 升序 或 降序 排序 方式 ， 默 认 值 为 ASC。 


外 说 明 : 在 给 数据 表 中 添加 索引 时 ， 索 引 通 常 是 以 IX 开头 的 。 


10.2.2 试 着 创建 聚集 索引 


聚集 索引 几乎 在 每 张 数据 表 都 存在 ， 读 者 这 时 会 想 了 ， 如 果 不 创建 索引 ， 也 会 存在 聚 
集 索 引 吗 ? 是 这 样 的 , 如 果 一 张 表 中 有 了 主键 , 那么 系统 就 会 认为 主键 列 就 是 聚集 索引 列 。 
为 了 让 读者 更 好 地 理解 索引 的 使 用 ， 先 为 本 章 创建 要 使 用 的 数据 库 和 数据 表 。 本 章 使 用 的 
数据 库 命 名 为 chapter10， 在 其 数据 库 中 创建 课外 辅导 班 课程 信息 表 (courseinfo)， 表 结构 
如 表 10-1 所 示 。 


表 10-1 课外 辅导 班 课程 信息 表 (courseinfo) 


编号 字 段 名 说 明 
1 id 编号 
区 name varchar(20) 辅导 班 名 称 
$ address varchar(30) 辅导 班 地 址 
4 tel varchar(15) 电话 号 码 
5 coursename varchar(20) 课程 名 称 〈 主 要 课程 ) 
6 price 课程 价格 (平均 价格 ) 
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按照 上 面 的 表 结构 创建 数据 表 ， 语 名 如 下 : 


USE chapter10; 
CREATE TABLE courseinfo 


( 

id nk 

name varchar (20), 
address varchar (30), 

七 el Varchar (15), 
coursename Varchar (20) , 
price int 

); 


执行 上 面 的 语句 ， 即 可 在 数据 库 chapter10 中 创建 数据 表 courseinfo。 有 了 数据 表 ， 
在 就 可 以 为 其 创建 索引 了 。 

【示例 1】 为 课外 辅导 班 课程 信息 表 (courseinfo〉 中 的 编号 (id) 列 创建 聚集 索引 

根据 题目 要 求 ， 使 用 CREATE UNIQUE CLUSTERED INDEX 语句 创建 聚集 索引 ， 
体 语句 如 下 : 

USE chapter10; 


CREATE UNIQUE CLUSTERED INDEX IX_ COURSEINFO _ID 
ON courseinfo (id); 


现 


o 


共 


执行 上 面 的 语句 ,就 可 以 为 表 courseinfo 中 的 id 列 创建 聚集 索引 了 。 执 行 效果 如 图 10.1 


所 示 。 
Ricrosoft SQL Server Wanagenent Studio =Iolxl 
文件 @) ”编辑 下 ) 视图 WW) 查询 @) 项 目 @) 调试 @) 工具 CD) 窗口 
社区 CC) 帮助 由 
辽 .新建 查询 四 | 鹿 | 疗 六 咏 | 让 | 芒 回 马 | 加 局 
; My Wa | chepter10 -| 执行 W) 有 v 3 避 回 | 3 攻 
SQLQueryl. s. . .rator (53))*| vx 


日 usF chapter10 
由 CREATE UNIQUE CLUSTERED INDEX IX_COURSEINFO_ID 
| courseinfo (1id); 


4 4 
已 消息 | 
命令 已 成 功 完成 。 - 司 
可 
|27\NSSQLSERVER2008 〔 . ，| YITDY-952T\Adninistrato .. | chapterl0 | 00:00:00 |0 行 


就 绪 行 5 列 1 Ch 1 
图 10.1 创建 聚集 索引 IX_COURSEINFO ID 


从 图 10.1 中 ， 就 可 以 看 出 索引 已 经 在 表 courseinfo 中 创建 成 功 了 。 那 么 ， 如 何 使 月 
句 查 看 到 为 表 创 建 的 索引 呢 ? 很 简单 , 使 用 系统 存储 过 程 SP_HELPINDEX 就 可 以 查看 
查看 为 表 courseinfo 创建 的 索引 ， 语 句 如 下 : 


SP_HELPINDEX 'courseinfo'; 


昌 语 


a 


执行 上 面 的 语句 ， 就 可 以 查看 到 表 courseinfo 中 创建 过 的 索引 了 。 目 前 ， 在 表 中 只 创 


建 了 一 个 索引 。 执 行 效果 如 图 10.2 所 示 。 


从 图 10.2 中 ， 就 可 以 清楚 地 看 到 为 表 courseinfo 创建 的 索引 的 类 型 以 及 该 索引 所 在 的 


列 名 。 
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Amicrosoft SQL Server Wanagenent Studio 
文件 全 编辑 EE) 视图 查 淘 @) 项 目 E) 调式 0) 工具 CD) 
窗口 WD 社区 加 帮助 中 


;也 汗 奸 查询 四 | 出 | 铠 巴 台 | 记 | 区 吕 己 | 怠 互 


=I9lxl 


; yq | chapter10 -| ?执行 DO 》 和 vv 品 富 | 国 
‘SQLQueryl. s.. .rator (53))#| vx 
[ 日 SP_HELPINDEX ‘COURSEINFO'; 
器 结果 | 忆 消息 | 


d 


[i pin 
区 .COURSEINFO ID § clustered, unique located on PRIMARY 


julsSQLSERVER2008 〔 .| YIDW-9527\Adninistrato. .. | chapter10 |00:00:00 |1 行 
就 绪 行 3 列 1 Ch1l a 


图 10.2 查看 courseinfo 表 中 的 索引 (示例 1) 


10.2.3” 试 着 创建 非 聚集 索引 


非 聚集 索引 在 一 张 数据 表 中 可 以 存在 多 个 ， 并 且 在 创建 非 聚集 索引 时 ， 可 以 不 将 其 列 
设置 成 唯一 索引 。 下 面 就 使 用 示例 2 来 演示 如 何 创建 非 聚集 索引 。 

【示例 2】 为 课外 辅导 课程 信息 表 〈courseinfo) 中 的 课程 名 称 〈coursename) 列 创建 
一 个 非 聚集 索引 。 

根据 题目 要 求 ， 使 用 CREATE NONCLUSTERED INDEX 语句 创建 非 聚集 索引 ， 有 具体 
语句 如 下 : 

USE chapter10; 


CREATE NONCLUSTERED INDEX IX _ COURSEINFO COURSENAME 
ON courseinfo(coursename); 


执行 上 面 的 语句 ， 就 可 以 为 courseinfo 表 中 的 coursename 列 创建 非 聚集 索引 了 。 执 行 
效果 如 图 10.3 所 示 。 


Microsoft SQL Server Nansgenent Stadio bs 
文件 和 ”编辑 FE) 视图 WD 查询 @) 项 目 E) 调式 W) 工具 中 
窗口 WD 社区 人 ) 帮助 0 
也 章 寻 查询 四 | 访 | 售 这 加 | 访 | 苞 回 马 | 加 避 
Wy 2 | chspterl0 *| ?执行 有 v ?3 可 回 | 
‘SQLQueryl. s.. -rater (53))#*| 3 
日 0SE chapter10; 

CREATE NONCLUSTERED INDEX IX COURSEINFO COURSENANE 

ON courseinfo (coursename); 


=I9|x| 


己 请 息 | 
命令 已 成 功 完成 。 
| 
INSSQLSERYER2008 (.. | WIDN-9527\Adninistrato. .. | chapterl0 |00:00:00 |0 行 
就 绪 行 4 列 1 Chi 


图 10.3 创建 非 聚 集 索引 


从 图 10.3 中 可 以 看 出 ,创建 非 聚集 索引 的 命令 执行 成 功 了 。 那 么 ， 使 用 系统 存储 过 程 
SP_HELPINDEX 查看 表 courseinfo 中 的 索引 ,看 看 是 否 已 经 存在 了 刚才 创建 的 非 聚 集 索引 
IX_COURSEINFO_COURSENAME。 查 询 效果 如 图 10.4 所 示 。 

从 图 10.4 中 可 以 看 出 ， 查 询 结果 中 的 第 1 行 就 是 新 创建 的 非 聚 集 索 引 IX_ 
COURSEINFO_COURSENAME。 


“Ms 
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ET 天 下 下 枯 
; My 和 | chapter10 -| 1 后 v 叶 富国 天 绚 


“SQLQueryl. s... rator (53))*| vx 
日 SP_HELPINDEX 'COURSEINFO'; 


nonclustered located on PRIMARY coursename 
clustered unique located on PRIMARY 。 这 


| TDy-s527WISsQLSERYER2008 〔 .，| WIDW-9527\Adninistrato. .. | chapterl0 |00:00:00 | 2 行 
就 绪 第 1 行 第 1 列 


图 10.4 查看 courseinfo 表 中 的 索引 (示例 2) 


10.2.4” 试 着 创建 复合 索引 


所 谓 复合 索引 ， 就 是 指 在 一 张 表 中 创建 索引 时 ， 索 引 列 可 以 由 多 列 组 成 ， 有 时 也 被 称 
为 组 合 索引 。 读 者 可 以 回想 一 下 ， 创 建 主键 的 时 候 ， 是 不 是 就 学 习 过 一 个 主键 约束 可 以 由 
多 列 组 成 呢 。 回 忆 起 来 了 ， 就 好 办 了 ， 就 是 将 索引 列 的 括号 中 放置 多 个 列 名 就 可 以 了 ， 并 
且 ， 每 个 列 名 之 间 用 去 号 隔 开 即 可 。 另 外 ， 复 合 索 引 可 以 是 聚集 索引 也 可 以 是 非 聚 集 索引 。 
下 面 就 来 看 看 示例 3 吧 。 

【示例 3】 为 课外 辅导 班 课程 信息 表 (courseinfo) 中 的 地 址 address) 列 和 电话 (tel) 
列 创建 一 个 复合 索引 。 

根据 题目 要 求 ， 要 创建 一 个 复合 索引 ， 由 于 在 表 courseinfo 中 已 经 在 示例 1 时 为 其 创 
建 了 聚集 索引 ， 因 此 ， 这 里 只 能 为 courseinfo 表 创建 非 聚集 索引 了 。 创 建 索引 的 语句 如 下 ， 

USE chapter10; 


CREATE NONCLUSTERED INDEX IX COURSEINFO ADDRESS TEL 
ON courseinfo(address,tel); 


执行 上 面 的 语句 ， 就 可 以 为 表 courseinfo 创建 一 个 复合 索引 了 。 执 行 效果 如 图 10.5 


Aicroso ft SQL Server Nanagenent Studio -lolx| 
文件 中 ”编辑 时) 视图 WD 查询 @) 项 目 E) 调式 0) 工具 CD) 窗口 岂 ) 


社区 加 ”帮助 中 
辽 新 建 查询 四 | 让 | 这 遍 加 | 包 | 记 四 本 | 邓 囊 
Wy oR | chepterlo -| ?执行 Oh 后 她 富 国 | 沁 绚 


“SQLQueryl. s.. -rator (53))*| vx 
唱 USE chapter10; 

由 CREATE NONCLUSTERED INDEX IX_COURSEINFO_MDDRESS_TEL 
FON courseinfo (address, tel); 


| WIDN-9527\MISSQLSERVER2008 〔 . ，| WIDN-9527\Adninistrato. .. | chapterl0 |00:00:00 |0 行 
就 绪 行 5 列 1 Ch 1 


图 10.5 创建 复合 索引 
好 了 ， 到 这 为 止 复合 索引 已 经 创建 完成 了 ， 下 面 就 使 用 SP_HELPINDEX 系统 存储 过 
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程 来 查看 表 courseinfo 中 的 索引 吧 。 执 行 效果 如 图 10.6 所 示 。 


RNmicroseft SQL Server Nanagenent Stadio 有 -lolxl 
文件 到 ) 编辑 E) ”视图 WD) 查询 @) 项 目 E) 调式 @) 工具 CD) 窗口 

社区 加) 帮助 中 

了 四 | 记 | 也 也 马 | 记 | 谍 吕 本 | 骂 电 

My Wy | -hapterl0 -| 执行 QW) 有 品 加 | 国 | 下 强 名 
Slaneryl. s .rator (C53))*| -x 
日 sp_HELPINDEX courseinfo 习 


回 结果 | 消息 | 


WX COURSEINFO ID chustered, uniaue located on PRI.。 计 
| WIDW-9527\MSSQLSERYVER2008 (.. | WIDW-9527\Adninistrato. .. | chapter10 |00:00:00 |3 行 
就 绪 第 1 行 第 1 列 


图 10.6 查看 courseinfo 表 中 的 索引 (示例 3) 


从 图 10.6 所 示 的 查询 结果 中 ， 可 以 看 出 第 1 行 就 是 新 创建 的 复合 索引 ， 在 index_keys 
列 中 标明 了 该 索引 是 由 address 和 tel 列 组 成 的 。 


10.3 修改 索引 


在 上 一 节 中 已 经 学 习 了 如 何 创 建 索引 ， 其 实 索引 也 是 可 以 修改 的 。 但 是 ， 并 不 能 修改 
索引 中 的 全 部 内 容 ,并且 不 仅 可 以 使 用 语句 来 修改 索引 也 可 以 通过 企业 管理 器 来 修改 索引 。 


10.3.1 修改 索引 的 语 ; 


修改 索引 的 语法 与 创建 索引 的 语法 有 很 大 的 区 别 ， 请 读者 一 定 要 认真 查看 哦 ! 具体 的 
语法 结构 如 下 : 
ALTER INDEX index name 
ON 
{ 
[database name]. table or view name 
} 
{ [REBUILD] 
PRETH NN <rebulild indexr option> ea)I 
[DISABLE] 
[REORGANIZE] 
[ PARTITION = partition number ] 


中 : 

index_name 项 : 索引 的 名 称 。 
database_name 项 : 数据 库 的 名 称 。 

table or view_name 项 : 表 或 视图 的 名 称 。 
REBUILD 项 : 使 用 相同 的 规则 生成 索引 。 
DISABLE 项 : 将 索引 禁用 。 

REORGANIZE 项 : 指定 将 重新 组 织 的 索引 。 


DODODOOOOCO* 
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从 上 面 的 修改 语句 不 难看 出 ， 修 改 索引 只 是 对 原 有 索引 进行 禁用 、 重 新 生成 等 操作 ， 
并 不 是 直接 更 改 原 有 索引 的 表 和 列 。 


10.3.2 ”禁用 索引 


索引 有 时 是 好 东西 也 是 坏 东 西 ， 好 东西 无 非 就 是 指 提高 查询 的 效率 ， 但 有 时 在 一 张 数 
据 表 中 创建 多 个 索引 ， 也 会 造成 对 空间 的 浪费 。 因 此 ， 有 时 需要 将 一 些 没有 必要 的 索引 禁 
用 ， 当 然 需要 的 时 候 再 启用 索引 。 

【示例 4】 将 课外 辅导 班 课 程 信息 表 (courseinfo ) 中 的 IX_COURSEINFO_ 
COURSENAME 索引 禁用 。 

根据 题目 要 求 ， 要 将 其 索引 禁用 ， 具 体 的 语句 如 下 : 

USE chpater10; 

ALTER INDEX IX COURSEINFO COURSENAME 


ON COURSEINFO 
DISABLE; 


通过 上 面 的 语句 ， 就 可 以 将 索引 IX_COURSEINFO_COURSENAME 禁用 。 也 就 是 说 
当 查 询 courseinfo 表 时 该 索引 就 会 失效 。 执 行 效果 如 图 10.7 所 示 。 

当 用 户 希 望 使 用 该 索引 时 ， 使 用 启用 的 语句 启用 该 索引 即 可 。 启 用 该 索引 时 ， 只 需要 
将 上 面 语句 中 的 DISABLE 换 成 ENABLE。 

读者 是 否 会 思考 一 个 问题 呢 ? 如 何 知 道 在 一 个 数据 表 中 哪些 索引 是 禁用 的 哪些 索引 
是 可 以 使 用 的 呢 ? 答案 很 简单 ， 那 就 是 通过 视图 sys.indexes 查询 就 可 以 了 。 由 于 在 
sys.indexes 视图 中 的 列 数 众 多 ， 为 了 让 读者 可 以 一 目 了 然 地 看 到 结果 ， 可 以 只 查询 其 中 的 
索引 名 称 列 Cname) 和 索引 是 否 禁 用 列 (is_disabled)。 查 询 的 语句 如 下 : 


SELECT name,is disabled FROM sys.indexes; 


执行 上 面 的 语句 ， 就 可 以 查询 到 索引 是 否 被 禁用 了 。 查 询 效果 如 图 10.8 所 示 。 


Microsoft SQL Server Managenment St el 
了 Microsoft SQL Server Management studid | 文件 编 加 全 视图 (0 查 误 Q) 项 目 (6) ”调试 (0) 工具 (D 窗口 (ww) 
文件 (E) 编辑 6) ”视图 W 查询 (0) 项 目 B) 调试 D) 工具 (D SO WD 


窗口 () ”社区 (QO) 帮助 (HH) 
革新 建 查询 | 六 | 六 启 串 | 届 | 蕊 回 对 | 加 局 
Wy Wy2 | chapterlo -| 执行 Wj 》 昌 渡 申 久 
“5QLQuery1.sql r10 (sa (52))* > 
章 USE chapter10; 
由 ALTER INDEX IX COURSEINFO COURSENANE 
ON COURSEINFO 
DISABLE; 


也 新建 查询 六 | 鞍 镶 态 | 蕊 回 马 | 加 
"| 执行 WW 更 w ?9 本 


| 
命令 已 成 功 完 成 。 


可 | 加 下 x_coupsEINFO_ADDRESS_TEL 0 
园 引 lew-440z841576F (10.0RTM) | sa (52) | chapter10 | 00:00:00 | 0 行 加 查询 已 成 功 .| 18M-4402841576F (10.0RTM) | sa (52) | chapterl0 | 00:00:00 | 94 行 
就 绪 行 6 列 ! chi 及 第 1 行 第 1 列 


图 10.7 使 用 DISABLE 禁用 索引 图 10.8 查看 索引 是 否 被 禁用 
从 图 10.8 所 示 的 查询 结果 中 ， 可 以 看 出 只 有 名 为 IX_COURSEINFO_COURSENAME 


的 索引 的 is_disabled 列 的 值 是 1。 换 句 话 说 ， 如 果 is_disabled 列 的 值 是 1， 就 代表 了 该 索 
引 是 被 禁用 的 ， 相 反 ， 如 果 该 列 的 值 是 0， 就 代表 了 该 索引 是 启用 的 。 


二 
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所 谓 重 新 生成 索引 就 是 指 将 原来 的 索引 删除 再 创建 一 个 新 的 索引 。 重 新 生成 索引 的 好 
处 是 可 以 减少 获取 所 请 求 数据 所 需 的 页 读 取 数 ， 以 便 提高 磁盘 性 能 。 重 新 生成 索引 使 用 的 
是 修改 索引 语法 中 的 REBUILD 关键 字 来 实现 的 。 

【示例 5】 重新 生成 课外 辅导 班 课程 信息 表 (courseinfo〉 中 的 IX_COURSEINFO_ID 


索引 。 


根据 题目 要 求 ， 使 用 REBUILD 关键 字 重 新 生成 索引 ， 语 句 如 下 : 


USE chapter10; 


ALTER INDEX IX COURSEINFO ID ON courseinfo 


REBUILD; 


执行 上 面 的 语句 ， 就 可 以 将 索引 重新 生成 。 执 行 效果 如 图 10.9 所 示 。 


10.3.4 修改 索引 名 


索引 的 名 称 与 上 一 章 中 介绍 的 视图 一 样 ， 
都 是 可 以 修改 的 , 并 且 都 可 以 使 用 系统 存储 过 
程 sp_rename 来 完成 修改 操作 。 下 面 就 使 用 示 


例 6 来 演示 如 何 修改 索引 的 名 称 。 


【示例 6】 将 名 为 IX_COURSEINFO ID 
的 索引 ， 名 字 改 成 IX_ NEW_COURSEINFO 


ID。 


根据 题目 要 求 ， 修 改 索 引 名称 的 语句 


如 下 : 


RM Microsoft SQL Server Management Studig 
文件 忆 ”六 辑 (E) ”视图 (VW) 查询 (9) 项 目 (PB) 调试 D) 工具 (D 
宣 口 (WW) 社区 (QO 帮助 () 

和 2 新 可 光 WV 六 沁 这 古书 | 区 回 对 | 加 

“| 了 执 和 0 》 及 品 部 久 


WY 0 | chapter10 


回 查 .| IBM-4402841576F (10.0RTM) | sa (52) | chapter10 | 00:00:00 | 0 行 
就 绪 行 5 列 1 Chl 


图 10.9 重新 生成 索引 


Sp_rename 'IX COURSEINFO _ID',' IX NEW_ COURSEINFO _ID'; 


执行 上 面 的 语句 ， 索 引 的 名 称 就 更 改 完 成 了 。 效 果 如 图 10.10 所 示 。 


了、Microsoft SQL Server Management Studio 


文件 (E) 编辑 (E) 视图 凡 ， 查 询 (Q) 项 目 P) 调试 D) 工具 (D 窗口 (Ww) 社区 (CO 帮助 (中) 


辽 新 强 查 汐 | 由 | 侯 也 吕 | 记 | 区 回忆 | 进 瑟 


于 起 | chepterlo "|W av 史 名 国政 驹 | 的 央 扩 | 三 | 素 时 
SQLQuery3.sql -istrator (52))* | ee 
站 sp_rename 'courseinfo.IX COURSEINFO_ID','IX NEW COURSEINFO_ID','INDEX'; 习 
局 消息 | 
注意 : 更 改 对 象 名 的 任 一 部 分 都 可 能 会 破坏 脚本 和 存储 过 程 。 上 
可 » 
回 查询 已 成 功 执行 。 | WIDW-9527\MS5QLSERYER2008 (..，| WIDW-9527\Administrato,,，| chapter10 | 00:00:00 | 0 行 
就 绪 行 4 列 ! chl Is /4 


图 10.10 ”给 索引 重 命名 


全 注 意 : 在 给 索引 重 命名 时 ， 一 定 要 将 原来 的 索引 名 前 面 加 上 该 索引 所 在 的 表 名 ， 否 则 在 


数据 库 中 是 查找 不 到 的 。 
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10.4 删除 索引 
在 前 面 介 绍 索引 时 ， 就 提 到 过 索引 既 可 以 给 数据 库 带 来 好 处 也 会 造成 数据 库存 储 中 的 
浪费 。 因 此 ， 当 表 中 的 索引 不 再 需要 时 ， 就 需要 及 时 将 这 些 索引 删除 。 
10.4.1 删除 索引 的 语法 
索引 与 前 面 学 习 过 的 视图 一 样 ， 也 是 通过 DROP 语句 删除 的 。 具 体 的 语法 规则 如 下 : 


DROP INDEX 


index_name ON 


{ 
[ database name. [ schema name ] . | schema name. ] 
table or view name 

} 
| We | 
| [ owner name. ] table or view name.index name 
| | 
} 
其 中 ， 


口 index_name 项 : 索引 名 称 。 

口 database_name 项 : 数据 库 的 名 称 。 

口 schema_name 项 : 该 表 或 视图 所 属 架构 的 名 称 。 

口 table or_ view_name 项 : 与 该 索引 关联 的 表 或 视图 的 名 称 。 


10.4.2 ”删除 一 个 索引 


从 删除 索引 的 语法 中 ， 可 以 看 出 在 删除 索引 时 可 以 一 次 删除 1 到 多 个 索引 。 为 了 让 读 
者 更 好 地 掌握 删除 索引 的 方法 ， 在 本 小 节 中 先 以 示例 7 为 例 讲解 如 何 删除 一 个 索引 。 
【示例 7】 删除 索引 名 为 IX_NEW_COURSEINFO ID 的 索引 。 


USE chapter10; 
DROP INDEX IX NEW COURSEINFO ID ON dbo.courseinfo; 


执行 上 面 的 语句 ， 效 果 如 图 10.11 所 示 。 


6 
文件 E) ”编辑 (E) ”视图 (Y) 查询 (9) ”项目 (E) ”调试 D) 工具 (D 窗 D(W) 社区 (QO 


帮助 (H) 
和 2 新 尘 寺 VW) 目 记 也 瑟 记 | 蕊 加 要 过 天 
a 9 | chapter10 -| 执行 WW 有 到 剖 | 辐 洲 怠 
5QLQuery3.sql -istrator (52))*| vx 
日 0SE chapter10; = 
DROP INDEX IX NEW COURSEINFO ID ON dbo.courseinfo; = 
‘ | » 
已 消息 | 
命令 已 成 功 完成 。 -= 
| 六 
| WIDW-9527\MSSQLSERVER2008(... | WIDW-9527\Administrato..，| chapter10 | 00:00:00 | 0 行 
就 绪 行 4 列 1 chi ne 


图 10.11 删除 索引 IX_NEW_COURSEINFO_ID 
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10.4.3 ”同时 删除 多 个 索引 


有 了 上 一 小 节 删 除 单个 索引 的 基础 ， 删 除 多 个 索引 应 该 不 是 问题 了 。 在 删除 多 个 索引 
时 , 只 需要 把 多 个 索引 名 依次 写 在 DROP INDEX 后 面 即 可 。 下 面 就 使 用 示例 8 来 学 习 如 何 
同时 删除 多 个 索引 。 

【示例 8】 删除 索引 名 为 IXNEW_COURSEINFO ID 和 IX COURSEINFO_ 
COURSENAME 的 索引 。 

USE chapter10; 

DROP INDEX 


IX NEW COURSEINEFO ID ON dbo.courseinfo, 
IX_COURSEINFO COURSENAME ON dbo.courseinfo; 


执行 上 面 的 语句 ， 效 果 如 图 10.12 所 示 。 


Rs Microsoft SQL Server Management Studio = 上 将 
文件 (编辑 (E) 视图 (查询 (O) 项目) 调 坛 D) 工具 (D 窗口 (w) 社区 (O 
帮助 (H) 

凡尘) | 市 | 也 也 加 | 记 | 盛 加 弓 | 过 下 

My Wa | chapterlo -| 执行 WW》 sv 中 昌国 | 小 枫 名 
SQLQuery3.sql -istrator (52))*| A 


章 USE chapter10; 


IX NEW COURSEINFO_ID ON dbo.courseinfo, 
IX_COURSEINFO_COURSENANE ON dbo.courseinfo; 


| 
命令 已 成 功 完成 。 - 
| » 


|| WIDW-9527\MS5QLSERVER2008 (...| WIDW-9527\Administrato..，| chapter10 | 00:00:00 | 0 行 
就 绪 行 6 列 1 Chl mn A 


图 10.12 ”同时 删除 多 个 索引 


10.5 使 用 企业 管理 器 操作 索引 


通过 前 面 几 节 的 学 习 , 相信 读者 已 经 对 索引 的 使 用 有 所 了 解 了 。 但 是 , 面 对 复 杂 的 SQL 
语句 ， 有 些 读者 还 是 有 些 看 不 下 去 了 。 不 要 紧 ， 在 本 节 中 将 讲解 如 何在 企业 管理 器 中 轻松 
完成 对 索引 的 操作 。 


10.5.1 使 用 企业 管理 器 创建 索引 


创建 索引 的 语法 中 有 些 关 键 字 是 比较 难 记 的 ， 那 么 ， 在 企业 管理 器 中 创建 索引 就 省 去 
了 很 多 的 麻烦 。 下 面 就 通过 示例 9 演示 如 何在 企业 管理 器 中 创建 索引 。 请 读者 认真 学 习 创 
建 索 引 的 每 一 个 步骤。 

【示例 9】 使 用 企业 管理 器 创建 示例 1 中 的 索引 IX_COURSEINFO ID。 

在 企业 管理 器 中 ， 创 建 索 引 分 为 如 下 4 个 步骤 。 

(1) 在 企业 管理 器 中 , 展开 chapter10 数据 库 节 点 , 在 该 节点 下 展开 courseinfo 表 节 点 ， 
并 右 击 “ 索 引 ” 选 项 ， 在 弹出 的 右键 菜单 中 ， 选 择 “ 新 建 索引 ”选项 。 如 图 10.13 所 示 。 
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多. 者 建 未 引 


Ga 
与 凤 相 - 有 帮助 

图 到 

社 这 项 : Fes 

语句 列 2 李 

ES 3 吉村 四 ) [ 

裤 29 

EE su 可 

TE 
雪 3| 刍 列 四 
和 次 指 订 而 六。 | 光世 类 型 | 大 小 | 标识 和 NULL 值 于 加 四， 

开除 8 
下 秩 0 
有 

服务 器 

MTP-952TVISSNLSERIEE2008 

这 接 

NIDN-952T Wnini strator 

对 二 看 连接 性 

进度 
就 缚 
Ea | 
4 


图 10.13 ”新 建 索引 界面 


(2) 在 图 10.13 所 示 的 界面 中 ， 输 入 索引 名 称 并 选择 索引 类 型 。 这 里 ， 索 引 名 称 是 IX_ 
COURSEINFO ID， 索引 类 型 是 聚集 索引 。 
(3) 添加 索引 设置 的 列 。 在 图 10.13 所 示 的 界面 中 , 单 击 “ 添 加 ”按钮 , 弹出 如 图 10.14 
所 示 界 面 。 
ly 
选择 要 添加 到 索引 键 的 表 列 。 


varchar (15) 
coursename varchar (20) 


price 


= | | ww | 
图 10.14 选择 添加 索引 列 


在 图 10.14 所 示 界 面 中 ， 单 击 要 设置 成 索引 的 列 。 这 里 ， 将 id 选中 ， 并 单 击 “ 确 定 ” 
按钮 ， 即 可 完成 索引 列 的 添加 操作 。 


(4) 完成 前 3 步 操作 后 ， 单 击 图 10.13 所 示 界 面 中 的 “确定 ”按钮 ， 即 可 完成 索引 IX_ 
COURSEINFO_ID 的 创建 。 
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10.5.2 ”使 用 企业 管理 器 修改 索引 


在 本 章 的 10.3 节 中 ， 已 经 学 习 过 了 使 用 SQL 语句 来 修改 索引 ， 包 括 禁 用 索引 、 重 新 
生成 索引 以 及 修改 索引 名 的 操作 。 下 面 就 分 别 使 用 示例 10、 示 例 11 和 示例 12 来 演示 在 企 
业 管 理 器 中 如 何 禁用 索引 、 重 新 生成 索引 以 及 重 命名 索引 。 同 时 ， 读 者 也 会 体会 到 企业 管 
理 器 的 便利 。 

【示例 10】 禁用 示例 9 中 新 创建 的 索引 IX_ COURSEINFO ID。 

在 企业 管理 器 中 ， 禁 用 索引 分 为 如 下 步骤 。 

(1) 在 企业 管理 器 中 , 展开 chapter10 数据 库 节点 , 在 该 节点 下 展开 courseinfo 表 节 点 ， 
展开 “索引 ”节点 ， 在 其 节点 下 ， 右 击 名 为 区 _COURSEINFO_ID 的 索引 。 在 弹出 的 右 
键 菜 单 中 ， 选 择 “ 禁 用 ”选项 ， 出 现 图 10.15 所 示 界 面 。 


ly 


=l9lxl 
ET 


和 峭 蝶 十 名? 不 过 , 禁用 菏 集 索引 后 插 无 法 访问 基础 表 。 通 过 重新 生成 索引 可 以 


服务 器 
WIDW~9527 \NSSQLSERYER2006 


连接 
WIDN-9527 \Adnini strator 


都 查看 连接 属性 


就 绪 


Cw ] ™w | 
图 10.15 ”禁用 索引 IX_COURSEINFO ID 
(2) 在 图 10.15 所 示 界 面 中 ， 单 击 “ 确 定 ” 按 钮 ， 弹 出 图 10.16 所 示 的 对 话 框 。 


已 禁用 豆 集 素 引 TX_ COURSEINFO _ID 将 无 法 访问 基础 表 Yourseinfo” 
是 否 继续 ? 


四 ED se | sw | ww | 


图 10.16 禁用 索引 提示 对 话 框 
在 图 10.16 所 示 的 界面 中 ， 单 击 “ 是 ”按钮 ， 即 可 完成 禁用 索引 的 设置 操作 。 


“217% 


第 3 篇 数据库 使 用 进 阶 


【示例 11】 重新 生成 索引 IX_COURSEINFO _ID。 

在 企业 管理 器 中 ， 重 新 生成 索引 需要 以 下 步骤 完成 。 

(1) 在 企业 管理 器 中 , 展开 chapter10 数据 库 节点 , 在 该 节点 下 展开 courseinfo 表 节 点 ， 
展开 “索引 ”节点 ， 在 其 节点 下 ， 右 击 名 为 的 COURSEINFO_ID 的 索引 。 在 弹出 的 右 
键 菜单 中 ， 选 择 “ 重 新 生成 ”选项 ， 出 现 图 10.17 所 示 界 面 。 
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图 10.17 重新 生成 索引 界面 


(2) 在 图 10.17 所 示 界 面 中 ， 单 击 “确定 ” 按 钮 ， 即 可 重新 生成 该 索引 。 

【示例 12】 将 索引 IX_COURSEINFO ID 重新 命名 成 IX_COURSEINFO ID NEW。 

在 企业 管理 器 中 也 可 以 修改 索引 的 名 称 ， 通 过 一 个 步骤 就 可 以 完成 唆 。 在 企业 管理 器 
中 ， 展 开 chapter10 数据 库 节 点 ， 在 该 节点 下 展开 courseinfo 表 节 点 ， 并 展开 “索引 ”节点 ， 
在 其 节点 下 ， 右 击 IX_COURSEINFO ID 节点， 在 弹出 的 右键 菜单 中 ， 选 择 “ 重 命名 ” 选 
项 。 将 其 名 称 改 成 IX_COURSEINFO_ID NEW 即 可 。 

通过 上 面 的 3 个 示例 ， 读 者 已 经 对 在 企业 管理 器 中 修改 索引 有 所 掌握 了 ， 现 在 就 请 读 
者 将 前 面 示例 2 中 的 索引 修改 一 下 吧 。 


10.5.3 ”使 用 企业 管理 器 删除 索引 


删除 索引 是 比较 简单 的 一 种 操作 了 ， 但 是 无 论 删 除 什 么 ， 要 恢复 都 是 比较 困难 的 。 因 

一 定 要 慎重 哦 ! 在 前 面 讲解 使 用 SQL 语句 删除 索引 时 , 提 到 过 可 以 一 次 删除 多 个 索引 ， 
但 是 ， 使 用 企业 管理 器 只 能 一 次 删除 一 个 索引 。 虽 然 ， 只 能 一 次 删除 一 个 索引 ， 但 是 ， 删 
除 的 速度 却 很 快 哦 ! 下 面 就 以 示例 13 为 例 讲解 如 何在 企业 管理 器 中 删除 索引 。 

【示例 13】 删除 索引 IX_COURSEINFO ID NEW。 

删除 索引 是 很 简单 的 , 只 要 按照 索引 名 找到 索引 就 好 办 了 。 下面 就 来 看 看 具体 的 步骤 。 
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在 企业 管理 器 中 ， 展 开 chapter10 数据 库 节点 ， 在 该 节点 下 展开 courseinfo 表 节 点 ， 并 
展开 “索引 ”节点 ， 在 其 节点 下 ， 右 击 X_ COURSEINFO_ ID NEW 节点 ， 在 弹出 的 右键 
菜单 中 ， 选 择 “ 删 除 ” 选 项 。 出 现 图 10.18 所 示 界 面 。 
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图 10.18 ”删除 索引 界面 
在 图 10.18 所 示 界 面 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 将 索引 IX_ COURSEINFO ID NEW 
删除 了 。 
10.6 本 章 小 结 


本 章 主 要 讲解 了 索引 的 分 类 、 创 建 、 修 改 以 及 删除 的 操作 。 在 索引 分 类 中 ， 主 要 讲解 
了 索引 中 的 聚集 索引 和 非 聚 集 索引 的 作用 ;在 创建 索引 中 ， 除 了 讲解 创建 索引 的 语法 外 ， 
还 分 别 演示 了 聚集 索引 、 非 聚集 索引 以 及 复合 索引 的 创建 ， 在 修改 索引 时 ， 着 重 讲解 了 如 
何 禁用 索引 、 重 新 生成 索引 以 及 重 命名 索引 ;在 删除 索引 时 ， 讲 解 了 一 次 删除 1 个 或 多 个 
索引 。 最 后 ， 还 讲解 了 索引 如 何在 企业 管理 器 中 使 用 。 


10.7 本 章 习题 


一 、 填 空 题 


1. 索引 可 以 分 为 ”  。 
2. 禁用 索引 的 关键 字 是 ” ”__。 
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3. 删除 索引 使 用 的 关键 字 是 。 
二 、 选 择 题 
1. 关于 索引 下 列 叙 述 正确 的 是 本 
A. 在 一 张 表 中 可 以 有 多 个 聚集 索引 和 非 聚 集 索引 
B. 在 一 张 表 中 只 能 有 一 个 聚集 索引 
C. 在 一 张 表 中 只 能 有 一 个 非 聚 集 索引 
D. 以 上 都 对 
2. 下 面 对 索 引 的 操作 描述 正确 的 是 5 
A. 索引 不 能 删除 
B. 使 用 语句 一 次 只 能 删除 一 个 索引 
C. 使 用 语句 一 次 可 以 删除 多 个 索引 


D. 以 上 都 不 对 
3. 重新 生成 索引 使 用 的 关键 字 是 

A. reuse B. renew C. rebuild 
三 、 问 答题 


1. 索引 的 作用 是 什么 ? 
2. 聚集 索引 和 非 聚 集 索引 的 区 别 是 什么 ? 
3. 如 何 创建 复合 索引 ? 


、 操 作 题 


对 于 课外 辅导 班 课程 信息 表 完 成 下 列 索引 操作 。 

(1) 给 课外 辅导 班 课 程 信息 表 的 id 列 创建 聚集 索引 。 
(2) 禁用 (1) 中 创建 的 索引 。 

(3) 重新 生成 (1) 中 的 索引 。 

(4) 将 (1) 中 创建 的 索引 更 名 。 

(5) 将 (1) 中 创建 的 索引 删除 。 


“a 


D. 以 上 都 不 对 
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TSQL 中 的 T 是 Transact 的 缩写 ， 它 是 在 标准 SQL 基础 上 改进 的 在 SQL Server 数据 

库 中 使 用 的 SQL 语言 。 在 TSQL 中 , 集合 了 ANSI89 和 ANSI92 标准 ， 并 在 此 基础 上 对 其 
扩展 。 因 此 ，T-SQL 并 不 适用 于 所 有 的 数据 库 ， 仅 可 以 在 SQL Server 中 使 用 。 打 个 比方 ， 
SQL 可 以 比 作 是 普通 话 ， 而 工 SQL 就 是 方言 ， 只 能 在 某 个 地 区 使 用 。 
章 的 主要 知识 点 如 下 : 

了 解 工 SQL 语法 规则 

什么 是 变量 和 常量 

如 何 使 用 流程 控制 语句 

如 何 使 用 游标 

如 何 使 用 事务 


OOOODO 


11.1 了 解 工 SQL 语法 规则 


读者 看 到 本 节 的 标题 可 能 会 有 些 糊涂 ， 尤 其 是 学 习 过 一 些 编程 语言 的 读者 ， 只 知道 每 
一 门 编程 语言 在 刚 开始 的 时 候 会 学 习 其 语法 规则 , 难道 SQL 语句 还 有 语法 规则 ? 答案 是 肯 
定 的 ,实际 上 ,TSQL 的 语法 规则 有 些 也 是 与 编程 语言 中 的 语法 规则 类 似 的 。 那 么 ,在 工 SQL 
中 通常 包括 哪些 基本 语法 呢 ? 通常 会 在 工 SQL 中 使 用 的 是 变量 、 常 量 以 及 流程 控制 语句 。 
下 面 就 来 一 一 对 其 介绍 如 下 。 

(1) 使 用 常量 和 变量 

在 前 面 的 章节 中 虽然 也 使 用 过 SQL 语句 ， 但 是 这 些 SQL 语句 都 是 通过 单一 的 一 条 语 
句 就 完成 某 一 个 操作 ， 比 如 : 向 数据 表 添 加 一 条 数据 ， 使 用 INSERT 语句 ;删除 数据 表 中 的 
一 条 数据 ， 使 用 UPDATE 语句 。 常 量 和 变量 通常 不 会 在 上 面 说 过 的 SQL 语句 中 出 现 ， 而 是 
在 一 个 或 多 个 语句 块 中 使 用 。 所 谓语 句 块 ， 就 是 由 多 条 SQL 语句 组 成 的 一 组 SQL 语句 。 

(2) 流程 控制 语句 

所 谓 流程 控制 语句 ， 就 是 指 用 来 控制 执行 语句 的 先后 顺序 的 ， 并 且 还 能 够 按 一 定 的 条 
件 来 控制 执行 哪些 SQL 语句 。 在 SQL 中 的 流程 控制 语句 ， 与 常见 的 编程 语言 ， 如 C# 语 言 
或 Java 语言 使 用 方法 类 似 。 此 外 ，SQL 中 也 有 捕获 异常 的 语句 。 


11.2 ”常量 和 变量 


常量 也 称 为 文字 值 或 标量 值 ， 是 表示 一 个 特定 数据 值 的 符号 。 常 量 的 格式 取决 于 它 所 
表示 的 值 的 数据 类 型 ， 在 对 于 数据 的 操作 中 ， 常 量 被 经 常 使 用 。 例 如 ,在 SELECT 语句 中 ， 
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可 以 使 用 常量 构建 查询 条 件 。 变 量 则 是 相对 于 常量 而 说 的 ， 变 量 的 值 是 可 以 改变 的 ， 通 常 
会 设置 一 个 标识 符 来 存储 变量 。 
11.2.1 常量 


在 SQL Server 中 ， 所 有 基本 的 数据 类 型 表示 的 值 ， 都 可 以 作为 常量 来 使 用 。 常 量 主要 
包括 字符 串 常量 、 二 进 制 常量 、 日 期 时 间 常 量 、 整 型 常量 和 数值 型 常量 等 。 下 面 就 依次 对 
这 几 种 类 型 的 常量 举例 说 明 。 

1. 字符 串 类 型 常量 


字符 串 常 量 包含 字母 、 数 字 字符 以 及 特殊 字符 ， 如 感叹 号 (!)、at 符 〈@) 和 井 字 号 
(#)。 此 外 ， 字 符 串 常量 还 要 用 单 引号 括 住 。 下 面 就 来 见识 见识 字符 串 常 量 : 


'today' 
'2012@126.com' 


2. 二 进 制 常量 

所 谓 二 进 制 常量 ， 就 是 用 二 进 制 数 表示 的 数 。 在 SQL Server 数据 库 中 ， 二 进 制 数 前 级 
是 0x。 这 部 分 常量 不 必 使 用 单 引 号 括 住 。 该 常量 不 仅 可 以 表示 具体 的 数值 也 可 以 表示 二 进 
制 字 符 串 。 下 面 就 来 看 看 二 进 制 常量 的 例子 吧 。 

0xmorning 


0x12E 
0x “( 空 二 进 制 字符 ) 


3. 日 期 时 间 常 量 


日 期 时 间 常 量 是 由 datetime 类 型 的 数据 组 成 的 。 该 常量 要 使 用 特定 形式 的 字符 日 期 值 
来 表示 ， 并 且 也 要 用 单 引号 括 起 来 。 日 期 时 间 常 量 格式 有 很 多 ， 读 者 可 以 参考 本 书 第 3 章 
数据 类 型 部 分 的 内 容 。 下 面 就 来 看 看 如 何 表示 日 期 时 间 类 型 的 常量 。 

123 May, 2012' 

'20120101' 

'06/1/2012' 

os 0 

'05:24 PM' 

从 上 面 的 例子 可 以 看 出 ， 前 3 个 例子 表示 的 是 日 期 类 型 的 常量 ， 而 后 面 的 2 个 例子 表 
示 的 是 时 间 类 型 的 常量 。 不 论 是 哪 种 类 型 的 常量 ， 读 者 都 要 按照 日 期 时 间 类 型 数据 的 格式 
来 写 ， 并 且 一 定 要 用 单 引号 括 住 哦 ! 


4. 整 型 常量 


整 型 常量 ， 应 该 是 读者 最 为 熟悉 的 一 种 常量 了 。 所 谓 整 型 常量 就 是 指 不 包含 小 数 点 的 
数 。 此 外 ， 该 常量 也 不 必用 单 引 号 括 住 。 既 然 如 此 简单 ， 下 面 就 来 见识 见识 吧 。 
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5. 数值 型 常量 


数值 型 常量 的 表示 范围 就 要 比 整 型 常量 表示 的 范围 要 广泛 一 些 ， 不 仅 包 含 整数 也 包含 
小 数 。 但 是 ， 无 论 是 整数 还 是 小 数 ， 也 不 需要 使 用 单 引 号 将 其 括 住 。 下 面 就 看 看 数值 型 常 
量 的 例子 吧 。 

pih ep 

Ey 

通过 上 面 对 5 种 主要 常量 的 讲解 , 现在 就 来 将 这 些 常 量 应 用 到 T-SQL 语言 中 来 尝试 一 
下 。 在 讲解 示例 1 之 前 ， 需 要 读者 先 创建 本 章 使 用 的 数据 库 chapter11。 

【示例 1】 在 数据 库 chapter11 中 ,创建 商品 信息 表 productinfo 来 存放 商品 信息 ， 表 结 
构 如 表 11-1 所 示 。 并 在 此 表 中 存放 如 表 11-2 所 示 的 数据 。 使 用 T-SQL 语句 ， 将 办 公 类 商 
品 的 价格 上 调 10 元 。 


表 11-1 商品 信息 表 (productinfo) 


编号 数据 类 型 说 明 
1 int 编号 
2 varchar(20) 商品 名 称 
3 decimal(6,2) 商品 价格 
4 varchar(20) 商品 类 型 
5 varchar(15) 商品 产地 
6 varchar(15) 厂商 电话 


表 11-2 ”商品 信息 表 中 的 数据 
编号 厂商 电话 
1 010-12345678 
2 024-12345678 
3 


021-12345678 


有 了 数据 表 和 数据 ， 现 在 就 可 以 来 编写 SQL 语句 了 ， 具 体 语 句 如 下 : 


USE chapterll; 
UPDATE productinfo SET price=price+10 WHERE type=' 办 公 '; 


执行 上 面 的 语句 ， 就 可 以 将 商品 信息 表 中 价格 列 的 信息 更 新 了 。 这 是 
是 我 们 使 用 的 常量 呢 ? 没 错 ， 就 是 10 这 个 整 型 常量 。 


全 说 明 : 常量 不 仅 可 以 放置 在 SET 语句 中 ， 可 以 在 SQL 语句 中 的 任何 语句 里 出 现 ， 并 且 
可 以 参与 计算 。 但 是 ， 在 计算 时 还 要 注意 数据 类 型 的 转换 哦 ! 


员 


， 读 者 看 看 哪个 


11.2.2 ”变量 


T-SQL 语句 中 的 变量 主要 应 用 在 后 面 要 讲解 的 流程 控制 语句 中 。 变 量 主要 包括 局 部 变 
量 和 全 局 变量 。 局 部 变量 是 指 用 户 自 定义 的 变量 ， 而 全 局 变量 是 指 SQL Server 数据 库 中 系 
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统 自 带 的 一 些 变 量 。 下 面 就 分 别 来 讲解 局 部 变量 和 全 局 变量 的 使 用 。 


T-SQL 局 部 变量 必须 要 先 声 明 后 使 用 ， 并 且 在 声明 时 要 指定 变量 的 数据 类 型 ， 声 明 数 
据 类 型 后 ， 该 变量 就 只 能 存在 该 类 型 的 数据 了 。 下 面 就 来 看 看 局 部 变量 是 如 何 声明 和 赋值 
的 吧 。 

(1) 声明 局 部 变量 

在 TSQL 中 ， 局 部 变量 使 用 DECLARE 关键 字 来 声明 ， 并 且 可 以 一 次 声明 多 个 变量 。 
特别 需要 注意 的 是 ， 局 部 变量 名 前 都 要 加 上 前 级 @。 具 体 的 语法 形式 如 下 : 


DECLARE @var name datatype, @var name datatype,..; 


其 中 : 

口 @var_name: var_name 是 变量 名 ，@ 是 局 部 变量 的 前 绥 。 
口 datatype: 数据 类 型 。 该 数据 类 型 是 系统 内 置 的 数据 类 型 。 
下 面 就 举 个 简单 的 例子 看 看 变量 是 如 何 定义 的 。 


DECLARE @name varchar (20), Q@age int; 


上 面 的 语句 分 别 定义 了 一 个 字符 串 类 型 变量 @name 和 一 个 整 型 变量 @age。 

(2) 给 变量 赋值 

知道 了 变量 是 如 何 声 明 的 ， 现 在 要 学 习 的 是 如 何 给 这 些 变量 赋值 。 局 部 变量 的 赋值 通 
常 有 两 种 方法 , 一 种 是 使 用 SET 关键 字 赋值 ， 一 种 是 使 用 SELECT 关键 字 赋 值 。 下 面 就 来 
看 看 具体 的 语法 说 明 吧 。 


SET @var name=value; 


其 中 : 
口 @var_name: var_name 是 变量 名 ， 必 须 是 在 前 面 已 经 声明 过 的 变量 名 
口 value: 给 变量 赋 的 值 。 该 值 一 定 要 与 变量 的 数据 类 型 匹配 。 


SELECT @var name=value, @var name=value, 


通过 上 面 的 语法 可 以 看 出 , 使 用 SET 和 SELECT 都 可 以 为 局 部 变量 赋值 , 并且 赋 值 的 
方法 很 相似 。 但 是 ， 它 们 也 是 有 区 别 的 。 使 用 SET 关键 字 对 局 部 变量 赋值 ， 一 次 只 能 给 一 
个 变量 赋值 ， 而 使 用 SELECT 关键 字 对 局 部 变量 赋值 ， 一 次 可 以 给 多 个 变量 赋值 。 

下 面 就 用 示例 2 来 演示 如 何在 工 SQL 语句 中 使 用 局 部 变量 。 

【示例 2】 分 别 定义 商品 名 称 和 价格 的 变量 ， 并 给 其 赋值 。 

DECLRRE name varchar (30), @price decimal (6，2) 

SET ename=' 登 山 包 ' 

SET @price=305.5; 

执行 上 面 的 语句 ， 就 可 以 完成 声明 变量 和 对 变量 赋值 的 操作 了 。 这 里 ， 使 用 的 是 SET 
语句 对 变量 赋值 ， 请 读者 练习 使 用 SELECT 语句 替换 SET 语句 来 完成 示例 2。 执 行 效果 如 
图 11.1 所 示 。 
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oy 
文件 昌 ” 编 辑 ( 唱 ”视图 查询 (9) 项 目 (P) 调式 0) 工具 站 窗 D(w) 社区 (O 
帮助 (H) 


卫 彰 建 查 帮 | 六 | 二 六 名 | 谣 | 芒 回 马 | 妆 5 
a | depterll "| 执 和 5 》 mv 孙 可 国政 马 名 
SQLQueryl.sql -istrator (52))*| Xx 


me varchar (30), Bprice decimal (6, 2 


BB 


消息 | 
命令 已 成 功 完成 FP 
可 


|| WIDW-9527\MSSQLSERYER2008 (,.. | WIDW-9527VAdministrato.,.| chapterll | 00:00:00 | 0 行 
就 绪 行 5 列 1 Chl mn 


图 11.1 局 部 变量 的 声明 和 赋值 
全 说 明 : 如 果 想 查看 一 下 赋值 后 变量 的 值 ， 可 以 通过 PRINT 语句 将 其 值 输出 。 也 可 以 直 


接 使 用 SELECT 语句 来 显示 变量 值 。 要 显示 示例 2 中 变量 @name 的 值 ， 可 以 使 
用 以 下 方法 。 
PRINT @name; 


SELECT Q@name; 


2. 全 局 变量 


全 局 变量 是 系统 自 带 的 变量 , 不 需要 定义 就 可 以 直接 使 用 。 在 SQL Server 数据 库 中 ， 
全 局 变量 是 以 @@ 为 前 级 的 。 常 用 的 全 局 变量 如 表 11-3 所 示 。 
表 11-3 常用 的 全 局 变量 
说 有明 
存储 上 一 次 执行 语句 的 错误 代码 
存储 最 后 插入 行 的 标识 列 的 值 
存储 数据 库 的 版 本 信息 
存储 上 一 次 执行 语句 影响 的 行 数 
存储 上 一 次 FETCH 语句 的 状态 值 


见识 了 表 11-3 所 示 的 全 局 变量 , 那么 , 这些 全 局 变量 如 何 使 用 呢 ? 下 面 就 用 示例 3 来 
演示 如 何 使 用 全 局 变量 。 

【示例 3】 通过 全 局 变量 查看 当前 数据 库 的 版 本 信息 。 

显示 版 本 信息 所 用 的 全 局 变量 是 @@VERSION， 具 体 的 语句 如 下 : 

SELECT Q@@VERSION; 

执行 效果 如 图 11.2 所 示 。 

通过 图 11.2 所 示 的 界面 可 以 看 出 , 全 局 变量 直接 在 SELECT 语句 中 使 用 就 可 以 查看 出 
结果 ,全 局 变量 除了 在 SELECT 语句 中 使 用 , 还 经 常用 在 下 一 节 所 讲解 的 流程 控制 语句 中 。 
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本 Microsol ft SQL Server Management Studio =Iolx|l 
文件 (FE) ”编辑 视 攻 (YW) ”查询 (9) 项 目 P) 调试 D) 工具 (TD 
窗口 (W) 社区 (OO 帮助 (H) 


了 WE EE 
BH2 | chapterll RA mA 


SQL QueryL.sql -istrator GE 
select BBVERSION; 


, 结果 | ee 
| 10.0.1600.22 (intel X86) Jul 92.. 


YER2008 (,,，| WIDW-9527\VAdministrato.,，| chapter11 | 00:00:00 | 1 行 
第 1 行 第 1 列 


图 11.2 全 局 变量 @@VERSION 的 使 用 


11.3 ”流程 控制 语句 


流程 控制 语句 是 T-SQL 语句 中 的 主要 组 成 部 分 ， 它 通常 会 用 在 后 面 要 学 习 的 存储 过 
程 、 触 发 器 等 数据 库 对 象 中 。T-SQL 中 的 流程 控制 语句 主要 包括 BEGIN...END 语句 、 
语句 、WHILE 语句 、CASE 语句 、WAITFOR 语句 以 及 异常 处 理 的 TRY...CATCH 语句 。 
在 本 节 中 读者 就 要 与 这 些 语句 一 一 见面 了 。 


11.3.1 BEGIN...END 语句 


BEGIN...END 语句 相当 于 是 程序 设计 语句 中 的 一 对 括号 ， 在 其 括号 中 存放 的 是 一 组 
T-SQL 语句 。 在 一 个 BEGIN...END 中 的 语句 ， 可 以 视 为 一 个 整体 。 虽然 BEGIN 和 END 
表示 的 含义 相当 于 是 一 对 括号 ， 但 是 绝对 不 能 用 括号 来 代替 ,它们 是 工 SQL 语句 中 的 关键 
字 。 有 具体 的 语法 形式 如 下 : 


BEGIN 
{ 
sql statement | statement block 


} 

END 

其 中 : 

口 BEGIN...END: 语句 关键 字 ， 它 允许 嵌 套 。 

口 {sql_statementlstatement_block} 项 : 指 任何 有 效 的 工 SQL 语句 或 语句 块 。 所 谓语 句 
块 ， 就 是 指 多 条 SQL 语句 。 


11.3.2 IF 语句 


IF 语句 ， 主 要 是 对 工 SQL 语句 进行 条 件 判 断 的 ， 是 使 用 最 频繁 的 语句 之 一 。 它 的 执行 
过 程 是 ， 如 果 满 足 正 条件， 则 执行 正 后 面 的 语句 ， 和 否则 就 不 执行 。 此 外 , 在 正 条 件 语句 
中 , 还 可 以 选用 ELSE 关键 字 , 作为 不 满足 正 条 件 时 , 要 执行 的 语句 。 这 就 好 像 完 成 了 “如 
果 今 天 是 星期 一 我 就 去 游泳 ， 否 则 我 就 去 上 课 ” 的 意思 表达 。 具 体 的 语法 如 下 : 


IF (Boolean expression) 


“a 
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BEGIN 

{ sql statement | statement block } 
END 
[ ELSE 
BEGIN 

{ sql statement | statement block } 
END ] 


其 中 : 


(1) Boolean expression: 必须 是 能 够 返回 TRUE 或 FALSE 值 的 表达 式 。 
(2) {fsql_statementlstatement block}: 任何 工 SQL 语句 或 语句 块 。 
在 上 面 的 语法 中 ，IF 后 面 的 小 括号 是 可 以 省 略 的 ， 但 是 建议 读者 不 要 省 略 ， 以 免 降 低 


程序 的 可 读 性 。 
下 面 就 用 示例 4 来 演示 如 何 使 用 正 语句 。 


【示例 4】 使 用 正 语句 判断 变量 的 值 是 否 为 偶数 ， 如 果 是 偶数 ， 输 出 “该 数 是 偶数 ”， 


否则 输出 “该 数 是 奇数 ”。 
根据 题目 要 求 ， 具 体 语句 如 下 : 
DECLARE @num int; 
SET @num=10; 
IE (@num%2= 0) 
BEGIN 
PRINT “' 该 数 是 偶数 ' ; 
END 
ELSE 
BEGIN 


PRINT ' 该 数 是 奇数 '; 
END 


执行 上 面 的 语句 ， 效 果 如 图 11.3 所 示 。 
11.3.3 ”WHILE 语句 


WHILE 语句 是 循环 语句 ,用 于 重复 执行 符合 条 件 的 SQL 
语句 或 语句 块 ， 只 要 满足 WHILE 后 面 的 条 件 ， 就 重复 执行 
语句 。 那么 , 读者 就 会 问 了 , 会 不 会 出 现 不 停 地 执行 WHILE 
中 的 语句 的 现象 呢 ? 当然 会 了 ， 把 这 种 一 直 重 复 执 行 的 语句 
称 为 死 循环 。 如 果 要 想 避 免 死 循环 的 发 生 ， 就 要 为 WHILE 
循环 设置 合理 的 判断 条 件 ， 并 且 可 以 使 用 BREAK 和 
CONTINUE 关键 字 来 控制 循环 的 执行 。 具体 的 WHILE 语句 
的 语法 形式 如 下 : 

WHILE (Boolean expression) 

BEGIN 


{ sql statement | statement block } 
END 


其 中 : 


Microsoft SQL Server Management SiS T=] 


-9527Whdministrato,，| chapterll | 00:00:00 | 0 行 


文件 (E) ”编辑 (E) ”视图 (Y) ” 查 放 (9) 项目 (P) 
调 坛 (0) 工具 (D 窗口 (WW) 社区 (QO) 帮助 中 


和 2 和 去 V | 由 也 也 下 | 应 | 写 吕 蝇 
Ry 8 | chapterll = 六 


' 该 数 是 偶数 ' 


IN 
"该 数 是 奇数 ' 


EE 行动 hl 久 


图 11.3 正 语 句 的 使 用 


口 Boolean expression: 必须 是 能 够 返回 TRUE 或 FALSE 值 的 表达 式 。 


口 fsql_statementlstatement block}: T-SQL 语句 或 语句 块 。 
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前 面 已 经 提 到 过 在 循环 中 可 以 使 用 BREAK 和 CONTINUE 来 控制 循环 的 执行 。 实 际 
上 ， 它 们 的 作用 就 是 跳出 循环 ， 也 是 避免 发 生死 循环 的 重要 手段 。 那 么 ， 就 来 看 看 它们 的 
作用 吧 。 

口 BREAK: 跳出 WHILE 循环 ， 使 WHILE 循环 终止 。 

口 CONTINUE: 结束 当前 的 WHILE 循环 ， 继 续 下 一 次 循环 。 

下 面 就 用 示例 5 来 演示 如 何 使 用 WHILE 语句 。 

【示例 S】 使 用 WHILE 循环 输出 1 一 10 的 数 ， 其 中 不 包括 5。 

根据 题目 要 求 ， 具 体 的 语句 如 下 : 


DECLARE Q@i int; -- 声 明 变量 

SET @i=0; -- 赋 值 

WHILE (@i<=9) --WHILE 循环 开始 
SET @i=@i+l; 

BEGIN 

IF (@i=5) 

BEGIN 

CONTINUE; -- 当 ei=5 时 结束 当前 循环 ， 继 续 下 一 次 循环 
END 

PRINT @i; 

END 


执行 上 面 的 语句 ， 效 果 如 图 11.4 所 示 。 


文件 (E) ”网 贺 (E) ”视图 VY) 查 汶 9) 项 目 (EP) 
调试 D) 工具 D 窗 DwW 社区 (OO Wh) 
了 5 亡 | 孔 也 可 | 户 | 蕊 品 中 
uy WR | chapterl1 "| ?执行 9 名 
SGuerylsq istrator (52))* | 

ine, 


DECLARE 81 im 


图 11.4 WHILE 语句 的 使 用 


从 图 11.4 所 示 的 输出 结果 ， 可 以 看 出 当 @i 是 5 的 时 候 ， 并 没有 将 其 值 输出 。 如 果 将 
CONTINUE 换 成 BREAK， 效 果 又 会 是 什么 样 的 呢 ? 请 读者 自己 来 试 试 吧 ! 


11.3.4 ”CASE 语句 


CASE 语句 与 下 类 似 ， 都 被 称 为 选择 语句 。 但 是 ，CASE 语句 不 同 于 IF 的 是 ，CASE 


“a 
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语句 可 以 设置 多 个 条 件 进行 判断 有 一 些 CASE 语句 是 可 以 直接 用 正 语句 来 转换 的 .CASE 
语句 的 基本 语法 如 下 : 
CASE input expression 
WHEN when expression THEN result expression 
Cen] 


[ 


ELSE else result expression 


END 


其 中 : 
口 input_expression: 条 件 ， 任 意 表达 式 。 
口 when_expression: 条 件 , 任意 表达 式 , 但 是 该 表达 式 的 结果 必须 要 与 input_expression 
口 result_expression: 当 input_expression=when_expression 的 结果 为 TRUE 时 返回 的 
表达 式 。 
口 else_result_expression: 当前 面 的 when_expression 条 件 全 都 不 满足 时 返回 的 表达 式 。 
这 里 ， 如 果 省 略 了 CASE 后 面 的 条 件 ， 那 么 ， 此 时 的 CASE 语句 就 被 称 为 搜索 式 的 
CASE 语句 。 但 是 ， 当 成 为 搜索 式 的 CASE 语句 后 ，WHEN 关键 字 后 面 的 表达 式 结果 就 必 
须 是 布尔 类 型 的 值 。 
下 面 就 用 示例 6 来 演示 如 何 使 用 CASE 语句 。 
【示例 6】 查询 商品 信息 表 (productinfo)， 并 使 用 CASE 语句 运算 ， 当 商品 的 类 型 是 
生活 类 时 ， 将 其 价格 涨 10 元 ， 当 商品 的 类 型 是 办 公 类 时 ， 将 其 价格 降 5 元 。 
根据 题目 要 求 ， 具 体 的 语句 如 下 : 
USE chapterll; 
SELECT name RS ' 商 品名 称 '，type RS ' 商 品类 型 ',price AS ' 原 来 的 商品 价格 ', ' 新 
商品 价格 '= CASE type 
WHEN ' 生 活 ' THEN price+10 
WHEN ' 办 公 ' THEN price-5 


END 
FROM productinfo; 


执行 上 面 的 语句 ， 效 果 如 图 11.5 所 示 。 


Microsolt SQLServer Management Sudio 
文件 E) 编辑 (E) ” 视 国 W 查询 (9) 项 目 (E) 调式 D) 工具 (D 窗口 (Ww) 社区 (QO 帮助 HD 
EL ME 


于 下马 | 镁 立 六 | 三 | 幸 地 /入 5s 


商品 名 称 | 南 品 关 型 | 原 末 的 商品 价格 | 新 商品 价格 | 
i ET a000 
2 | 水杯 生活 5000 6000 
se Ha 20 300 
回 查询 已 成 功 执 行 - | WIDW-9527\M55QLSERVER2008(...| WIDW-9527\Administrato...| chapterll | 00:00:00 | 3 行 
就 绪 行 9 到 1 Chl Mm 


图 11.5 ”CASE 语句 的 使 用 
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上 面 的 例子 中 ， 使 用 的 是 一 般 情况 下 的 CASE 语句 。 下 面 通过 示例 7 演示 如 何 使 用 搜 
索 式 的 CASE 语句 。 


【示例 7】 查询 商品 信息 表 (productinfo)， 并 使 用 CASE 语句 对 商品 通过 价格 分 类 。 
价格 大 于 50 元 ， 显 示 “ 高 价 商品 ” 当 价格 大 于 10 元 ， 显 示 “ 正 价 商品 ” 当 价 格 小 于 
10 元 时 ， 显 示 “ 促 销 商品 ”。 
根据 题目 要 求 ， 具 体 语 句 如 下 : 

USE chapterll; 


SELECT name RS ' 商 品名 称 '，type AS ' 商 品类 型 ',price AS' 商 品 价格 ',' 商 品 分 类 '= 
CASE 

WHEN price>=50 THEN ' 高 价 商品 ' 

WHEN price>=10 THEN ' 正 价 商品 ' 


WHEN price<10 THEN ' 促 销 商 品 ' 
END 
FROM productinfo; 


a 


执行 上 面 的 语句 ， 效 果 如 图 11.6 所 示 。 


< Microsoft SQL Seryer Ma 


文件 人 月 缉 E) 视图 W)， 查询 Go) 项 目 @) 请 D) 工具 D 窗 D(w) 社区 (O 帮助 

也 新建 查 j 让 | 这 孔 轧 记 | 蕊 台 司 | 过 下 

时 | depterll | 15W 》 mv 光 名 加 | 站 鸡 | 们 较 训 | 三 全 | 到 素 | 名 启 
SQLQuery1.sql -istrator (52))* | i 
BUvSE chapterll; = 
由 SELECT name As ! 商 品名 称 '，type Ms ' 商 品类 型 ' ,price Ms ' 商 品 价格 ',' 商 品 分 类 ' 


ChsE 
VEEN price>=50 THEN ' 高 价 商品 
WHEN price>~10 THEN ' 正 价 商品 ' 
WHEN price<10 THEN ,促销 商品 ， 
END 

上 PRom productinfo, 


办 公 200 


| WiDW-9527WM55QLSERVER2008 (,.。| WIDW-9527\Administrato, .| chapterll | 00:00:00 | 3 行 
行 10 列 1 chl 


图 11.6 ”搜索 式 CASE 语句 的 使 用 


We 


11.3.5 WAITFOR 语句 


WAITFOR 语句 可 以 控制 语句 执行 的 时 间 。 比如 : 1 分 钟 后 执行 语句 或 者 在 13 :00 执行 
语句 等 。 但 是 , 一 定 要 记 住 的 是 WAITFOR 语句 只 能 够 控制 24 小 时 之 内 的 时 间 范 围 。 具 体 
的 语法 形式 如 下 : 

WAITFOR 

{ DELAY 'time to pass' 

| TIME 'time to_ execute' 

} 

其 中 : 

口 time to_pass 项 : 等 待 多 长 时 间 可 以 执行 。 

口 time to_execute 项 : 设置 语句 的 执行 具体 时 间 。 

下 面 通过 示例 8 来 演示 如 何 使 用 WAITFOR 语句 。 


"0 
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【示例 8】 使 用 WAITFOR 语句 完成 下 列 操作 。 

(1) 在 10 秒 钟 后 ， 查 询 商品 信息 表 (productinfo) 中 的 商品 名 称 (name) 信息 。 

(2) 在 13:45 时 , 查询 商品 信息 表 (productinfo) 中 商品 名 称 Cname) 和 商品 价格 (price) 
的 信息 。 

根据 题目 要 求 ， 依 次 完成 下 列 操 作 。 

(1) 根据 题目 要 求 ， 要 在 10 秒 钟 后 执行 语句 ， 那 么 就 使 用 WAITFOR 语句 的 DELAY 
语句 完成 即 可 。 具 体 语 句 如 下 : 

USE chapterll; 

WAITFOR DELAY '00:00:10'; 

SELECT name FORM productinfo; 

执行 上 面 语句 ， 效 果 如 图 11.7 所 示 。 

(2) 根据 题目 要 求 ， 要 在 13:45 时 执行 语句 ， 那 么 ， 就 使 用 WAITFOR 语句 中 的 TIME 
语句 来 完成 。 具 体 的 语句 如 下 : 

USE chapterll; 


WAITFOR TIME "13:45:00'7 
SELECT name, price FORM productinfo; 


执行 上 面 的 语句 ， 效 果 如 图 11.8 所 示 。 


上 、Microsoft SQL Server Management St [= EE]| Microsoft SQL Server Management Studio =Iolxl 
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Sol Query sql RCI ~x 
USE chapterll: - 
WAITFOR TINE '13:45:00' 


SELECT name, price FRON productinfo. 


(S27\MS5QLSERVER2008 (,,，| WIDW-9527\Administrato,,，| chapterll | 00:00:16 | 3 行 
、 要 行 时 间 就 结 行 6 列 1 hi 及 


图 11.7 WAITFOR 语句 中 DELAY 语句 图 11.8 WAITFOR 语句 中 TIME 语句 的 使 用 


11.3.6 ”TRY...CATCH 语句 


TRY...CATCH 语句 是 捕获 异常 的 语句 。 学 习 过 C# 或 者 是 Java 语言 的 读者 ， 对 捕获 异 
常 一 定 不 陌生 了 ， 就 是 当 语句 出 现 错误 时 ， 来 处 理 错误 的 一 种 语句 。 但 是 ， 它 并 不 能 将 错 
误 修 改 ， 只 是 能 够 获得 一 些 错 误 信 息 。 下 面 就 来 认识 认识 TSQL 语句 中 TRY...CATCH 语 
名 的 语法 形式 。 


BEGIN TRY 

{ sql statement | statement block } 
END TRY 
BEGIN CATCH 

{ sql statement | statement block } 
END CATCH 
Wa 
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其 中 : 
口 sql_statement | statement_ block: 任何 工 SQL 语句 或 语句 块 。 
BEGIN TRY...END TRY: 在 TRY 之 间 的 语句 ， 是 可 能 会 发 生 异常 的 一 些 语句 。 
口 BEGIN CATCH...END CATCH: 在 CATCH 之 间 的 语句 ， 是 当 TRY 之 间 的 语句 出 
现 异常 时 执行 的 语句 。 通常 在 CATCH 语句 中 , 可 以 获取 到 相应 的 错误 号 以 及 错误 
信息 。 获 取 错 误 信 息 ， 可 以 使 用 的 函数 如 表 11-4 所 示 。 


表 11-4 ”获取 错误 信息 的 常用 函数 


口 


函 数 名 说 明 
ERROR NUMBERO 返回 错误 号 
ERROR STATE() 返回 错误 状态 号 


ERROR PROCEDURE() 
ERROR_LINEO 
ERROR MESSAGE() 


下 面 就 道 过 示例 9 来 演示 如 何在 TSQL 语句 中 捕获 异常 。 

【示例 9】 使 用 TRY.…CATCH 语句 捕获 异常 ， 向 商品 信息 表 中 插入 一 条 数据 ,商品 
号 列 (id) 插入 1， 并 显示 错误 号 和 错误 信息 。 

根据 题目 要 求 ， 向 商品 信息 表 中 插入 数据 ， 由 于 在 商品 信息 表 中 商品 编号 列 是 主键 ， 
因此 ， 再 插入 编号 是 1 的 数据 ， 就 会 出 现 错误 。 有 具体 的 语句 如 下 : 

USE chapterll; 

BEGIN TRY 


INSERT INTO productinfo (id,name,price)VALUES (1, ' 水 性 笔 ', 2); 
END TRY 

BEGIN CATCH 

SELECT 

ERROR_NUMBER () RS ' 错 误 号 '， 

ERROR_MESSAGE () RS ' 错 误 信息 '; 

END CATCH; 


执行 上 面 的 代码 ， 效 果 如 图 11.9 所 示 。 


[Rcroson SOL Somer onooement Sd Zly 
文件 EC) ”编辑 (E) ”视力 (V) 查询 9) ”项 目 (E) 调试 D) 工具 (D 窗口 (Ww) 社区 (CO 帮助 (H) 

了 新建 查 记 N | 让 | 也 也 可 记忆 四 与 | 马 忆 

aR | 中 hapterll "|? 执 有 v 引导 | 辐 | 了 动 | 礁 侧 | 耻 | 三 全 
SQLQuery1.sql -istrator (52))* 


返回 出 现 错误 的 存储 过 程 或 触发 器 名 称 
返回 导致 错误 的 例 程 中 的 行 号 
返回 错误 消息 的 内 容 


日 0SE chapterll F 
DBEGIN TRY 
INSERT INTO productinfo (id, name,price) VALUES (1, ' 水 性 笔 ' ,2) ; 
LEND TRY 
BEGIN CATCH 
an carca: -| 
1 * 厂 | 
品 结果 | 消息 | 
请 误 号 | 错误 信息 
1 违反 了 PRIMARY KEY 约束 PK_productinio'。 不 能 在 对 象 dbo productinfo 中 插入 重复 键 。 
回音 询 已 成 功 执行 。 。 | WIDW-9527WM55QLSERYER2008 (.，| WIDW-9527\Administrato,,，| chapterll | 00:00:00 | 1 行 


就 绪 第 1 行 第 1 列 Ins 


图 11.9 TRY...CATCH 语句 的 使 用 
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从 图 11.9 所 示 的 结果 可 以 看 出 , 通过 捕获 异常 就 可 以 很 容易 知道 语句 中 出 现 了 什么 问 
题 。 读 者 可 以 在 上 面 的 例子 中 ， 将 其 他 的 获取 错误 消息 函数 试 着 使 用 ， 并 查看 其 效果 。 


11.4 游 标 


用 户 在 数据 库 中 查询 数据 时 ， 查 询 出 的 结果 都 是 一 组 数据 或 者 说 是 一 个 数据 集合 。 如 
果 想 查看 其 中 的 某 一 条 数据 ， 只 能 通过 WHERE 条 件 语 句 来 控制 。 使 用 WHERE 语句 来 控 
制 的 方法 固然 简单 ， 但 是 ， 又 缺乏 灵活 性 ， 要 查看 每 条 数据 使 用 WHERE 语句 逐条 查询 就 
很 麻烦 了 。 为 了 改善 WHERE 语句 带 来 的 不 便 ,， 在 SQL Server 中 提供 了 游标 这 种 操作 结果 
集 的 方式 。 


11.4.1 


定义 游标 


游标 与 前 面 学 习 过 的 变量 一 样 ， 都 是 要 先 定义 再 使 用 的 。 定 义 的 方法 与 定义 变量 的 方 
法 类 似 ， 都 是 使 用 DECLARE 关键 字 。 具 体 的 语法 如 下 : 


DECLARE cursor name [ INSENSITIVE ] [ SCROLL ] CURSOR FOR select statement 
[ FOR { READ ONLY | UPDATE [ OF column name [ ,...n ] ]}] 


[;] 


其 中 : 


口 
口 


口 
口 
口 


cursor_name: 游标 的 名 称 。 遵 循 标识 符 定义 的 规则 。 

INSENSITIVE: 指定 创建 所 定义 的 游标 使 用 的 数据 临时 复 本 。 也 表明 该 游标 的 所 
有 请 求 均 在 tempdb 得 到 回应 ， 该 游标 不 允许 修改 。 

SCROLL: 指定 游标 的 提取 方式 (FIRST、LAST、PRIOR、NEXT、RELATIVE 或 
ABSOLUTE )。 

select_statement 项 : SELECT 语句 。 

READ ONLY: 禁止 通过 该 游标 进行 更 新 。 

UPDATE [OF column_name [,…n]]: 声明 游标 中 能 够 更 新 的 列 。 


11.4.2 ”打开 游标 
游标 与 其 他 数据 库 对 象 不 同 ， 不 仅 要 定义 游标 ， 还 要 在 使 用 游标 之 前 打开 游标 。 打 开 


游标 使 


OPEN 关键 字 。 具 体 的 语法 如 下 : 


OPEN { { [ GLOBAL ] cursor name } | cursor variable name } 


其 中 : 


口 
口 
口 


GLOBAL: 表示 该 游标 是 全 局 游标 。 
cursor_name: 游标 的 名 称 。 
cursor_variable_name: 游标 变量 的 名 称 。 


前 面 两 个 小 节 已 经 学 习 了 如 何 定义 和 打开 游标 ， 现 在 就 通过 示例 10 来 验证 学 习 效 果 。 
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【示例 10】 定义 游标 db_cursor 查询 商品 信息 表 (productinfo) 中 商品 名 称 (name) 
和 商品 价格 (price)。 并 使 用 OPEN 语句 打开 该 游标 。 
根据 题目 要 求 ， 具 体 语句 如 下 : 


USE chapterll; 
DECLARE db cursor scroll CURSOR FOR SELECT name, price FROM productinfo; 
OPEN db_ cursor; 


执行 上 面 的 语句 ， 效 果 如 图 11.10 所 示 。 


Be 
文件 (E) ”编辑 (E) ”视力 (W 查 淘 (Q) 项目 (P) 调式 (0) 工具 (D 窗口 (W) 社区 (QO 帮助) 

也 新 建 查询 由 六 这 也 一 记 | 写 加 本 驴 下 

My 1 | chapterll -| ?执行 0 有 vv 吕 加 国 ” 绚 | 的 央 | 术 | 三 全 名 
SQLQuery1.sql .istrator (52))* | 四 
日 0SE chapterll; 

DECLARE db_cursor scroll CURSOR FOR SELECT name,price FRONM productinfo; 
OPEN ab_curaor: 


己 消息 | 
命令 已 成 功 完成 。 

时 站 
园 坦 询 已 成 功 执 .| WIDW-9527WM55QLSERYER2008 (… | WIDW-9527WAdministrato,,，| chapterll | 00:00:00 | 0 行 
或 结 行 5 列 1 Chl 


图 11.10 ”声明 和 打开 游标 的 应 用 


11.4.3” 读 取 游 标 


读 取 游标 中 的 内 容 才 是 使 用 游标 的 重 中 之 重 。 读 取 游标 使 用 FETCH 关键 字 组 成 的 语 
句 来 完成 。 具 体 的 语法 形式 如 下 : 


FETCH 


[ [ NEXT | PRIOR | FIRST | LAST 
| ABSOLUTE n 
| RELATIVE n 


] 
FROM 


{ { [ GLOBAL ] cursor name } | @cursor variable name } 
[ INTO @variable name [ ,...n ] |] 


其 中 : 

口 NEXT: 表示 返回 结果 集中 当前 记录 的 下 一 条 记录 。 如 果 是 第 一 次 读 取 记录 则 返回 
的 是 第 1 条 记录 。 

口 PRIOR: 表示 返回 结果 集中 当前 记录 的 上 一 条 记录 。 如 果 是 第 一 次 读 取 记 录 则 不 
返回 任何 记录 。 

口 FIRST: 返回 结果 集中 的 第 一 条 记录 。 

口 LAST: 返回 结果 集中 的 最 后 一 条 记录 。 

口 ABSOLUTE n: 如 果 n 为 正 数 ， 则 返回 从 游标 中 读 取 的 第 n 行 记 录 ; 如 果 n 为 负 
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口 RELATIVE n: 如 果 n 为 正 数 ， 则 返回 从 当前 行 开始 的 第 n 行 记录 ; 如 果 n 为 负 
数 ， 则 返回 从 当前 行 开始 的 向 前 的 第 n 行 记录 。 
GLOBAL: 全 局 游标 。 
cursor_name: 游标 名 称 。 
@cursor_variable name: 游标 变量 名 。 
INTO @variable_name[,…n]: 将 提取 出 来 的 数据 存放 到 局 部 变量 中 。 

有 了 读 取 游标 的 语法 ， 就 已 经 完成 了 一 大 部 分 的 游标 学 习 任务 了 。 下 面 就 要 开始 演 
练 了 ! 

【示例 11】 创建 游标 db_cursor1， 查 询 商 品 信息 表 (productinfo) 中 的 商品 名 称 (name)、 
商品 价格 (price》 以 及 商品 产地 (address)。 并 使 用 FETCH 语句 读 取 游标 中 的 数据 。 

根据 题目 要 求 ， 具 体 的 语句 如 下 : 


USE chapterll; 
DECLARE db cursorl scroll CURSOR FOR SELECT name,price,address FROM 


口 
口 
口 
口 


productinfo; 
OPEN db _cursorl; -打开 游标 
FETCH NEXT FROM db _ cursorl -- 读 取 db_cursor1l 中 第 1 条 记录 
WHILE @@FETCH STATUS = 0 =-- 判 断 FETCH 命令 的 状态 
BEGIN 
FETCH NEXT FROM db cursorl =-- 向 下 逐条 读 取 db _cursorl 中 的 记录 
END 
执行 上 面 的 语句 ， 效 果 如 图 11.11 所 示 。 
二 glzxl 


文件 (EB) 编辑 (视图) 查询 Co) 项目) “调式 (0) 工具 (D 窗 Dw) 社区 (OO 玫 助 中 
了 We | 访 | 马 访 喇 | 让 | 上 日 马 | 如 电 
By 8 | chapterll -| ?1 扫 Fo 》 mm x 时 凶 加 | 站 鸡 | 冯 图 如 | 三 全 | 来 素 | 名 上 


5QLQuery1.sql .istrator (52))* | x 
日 0SE chapterll; 于 
DECLARE ob_cursorl scroll CURSOR FOR SELECT name,price,address FROM productinfo; 
OPEN db_cursc -- 打 开 游 标 
FETCH NEXT HE db_cursorl -- 读 取 db_cursor1 中 第 1 条 记录 
I WHILE BBFETCH STATUS = 0 -- 判 断 fetch 命 令 的 状态 
BEGIN 
FETCH NEXT FROM db_cursorl -- 向 下 逐条 读 取 db_cursor1 中 的 记录 
END 
= 
4 » 


加 查询 已 成 功 执行 。 | WIDW-9527\MS5QLSERYER2008 (… | WIDW-9527WAdministrato.,，| chapterll | 00:00:00 | 3 行 


行 12 列 1 Chi Ins 反 


图 11.11 使 用 FETCH 语句 查询 游标 中 的 数据 
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11.4.4 ”关闭 和 删除 游标 


读者 都 知道 数据 库 占用 内 存 ， 在 使 用 后 都 是 要 将 数据 库 关闭 的 。 实 际 上 ， 任 何 一 个 数 
据 库 对 象 都 是 占用 内 存 的 ， 游 标 也 不 例外 。 因 此 ， 在 游标 完成 了 特定 任务 后 ， 一 定 要 将 其 
关闭 。 如 果 游 标 以 后 都 不 会 再 用 了 ， 还 可 以 将 游标 删除 。 下 面 就 分 别 来 学 习 如 何 关闭 和 删 
除 游标 。 


1. 关闭 游标 
关闭 游标 后 ， 游 标 就 不 能 够 再 使 用 了 ， 也 不 能 够 再 读 取 游 标 中 的 内 容 了 。 但 是 ， 游 标 
是 可 以 重新 打开 的 。 关 闭 游标 的 具体 语句 如 下 : 


CLOSE {{[GLOBAL] cursor name} | cursor variable name} 


其 中 : 

口 GLOBAL: 全 局 游标 。 

口 cursor_name: 要 关闭 的 游标 名 称 。 

口 cursor_variable_name: 游标 变量 的 名 称 。 


2. 删除 游标 


删除 游标 使 用 的 可 不 是 DELETE 或 者 DROP 了 ， 删 除 游标 使 用 的 是 DEALLOCATE 
关键 字 。 删 除 后 的 游标 就 不 能 够 再 恢复 了 。 删 除 游 标的 语法 如 下 : 


DEALLOCATE {{ [GLOBAL] cursor name} | @cursor variable name} 


其 中 : 

口 cursor_name: 要 删除 的 游标 名 称 。 

口 @cursor_variable_name: 变量 的 名 称 。 

至 此 ， 对 游标 的 全 部 操作 就 已 经 学 习 完 了 。 下 面 就 通过 示例 12 来 一 同 应 用 所 学 的 全 
部 游标 操作 。 

【示例 12】 创建 游标 db_cursor2， 查 询 商 品 信息 表 (produceinfo〉 中 的 全 部 信息 。 并 
使 用 FETCH 读 取 数 据 ， 最 后 将 游标 先 关 闭 再 删除 。 

根据 题目 要 求 ， 具 体 的 语句 如 下 : 


USE chapterll; 
DECLARE db_cursor2 scroll CURSOR FOR SELECT * FROM productinfo; 


OPEN db_ cursor2; -打开 游标 

FETCH NEXT FROM db_cursor2 -- 读 取 db_cursor2 中 第 1 条 记录 
WHILE @@FETCH STATUS = 0 =-- 判 断 FETCH 命令 的 状态 

BEGIN 

FETCH NEXT FROM db_cursor2 -- 向 下 逐条 读 取 db_cursor2 中 的 记录 
END 

CLOSE db cursor2; -- 关 闭 游标 

DEALLOCATE db cursor2; =-- 删 除 游标 


执行 上 面 的 语句 ， 效 果 如 图 11.12 所 示 。 
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x Microsai ft SQL Server Management Studio 
文件 (E) ”编辑 (E) 视图 ( 几 。 查询 (9) ”项目 (EB) 调式 D) 工具 (D 窗口 (Ww) ”社区 (QO 帮助 (由 
站 V | 入 | 对 防 怨 | 入 | 芒 回 马 | 如 局 


村 8 | chapterll "| ?WwW 》 mv 加重 国 小 总 | 约 圈 名 | 三 
5QLQuery1.sql .istrator (52))* | 3 
日 USE chapterll; 三 

DECLARE db cursor2 scroll CURSOR FOR SELECT * FRON produccinfo: 

OPEN db_cursor2; -- 打 开 游标 

FETCH NEXT FROM db_cursor2 -- 读 取 db_cursor2 中 第 1 条 记录 

WHILE BBFETCH_STATUS = 0 ” -- 判 断 fecch 命 令 的 状态 


END 
CLOSE db_cursor2; 


一 关闭 游标 
DEALLOCATE db_cursor2; -- 荞 除 游 标 


回 结果 | 2 消息 | 
| nome [pice [vpe | aodess [tel | 
鼠标 3500 办 公 北京 010-12345678 


[|__| | name | pice | ype | addess | 由 
[2 水杯” 5000 生活 沈阳 02412345678 


BEGIN 
EE NEXT FRONM db_cursor2 -- 向 下 逐条 读 取 db_cursorz 中 的 记录 


(Jd | name | pice | ype | addess [ tel 
[3 | x 200 办 公 上 海 。 021-12345678 


多 查询 已 成 功 执 ，| WIDW-9527M55QLSERYER2008 (… | WIDW-9527\Administrato,.，| chapterll | 00:00:00 | 3 行 
就 绪 行 14 列 1 Chl ms 4 


图 11.12 ”游标 的 综合 应 用 


11.5 使 用 事务 控制 语句 


事务 可 以 看 作 是 一 件 具体 的 事 。 比 如 : 吃饭 、 看 电影 、 学 英语 等 。 在 SQL Server 数据 
库 中 ， 事 务 被 理解 成 是 一 个 独立 的 语句 单元 。 在 现在 的 生活 中 ， 已 经 习惯 了 按部就班 地 做 
每 件 事 ， 但 是 ， 一 旦 要 同时 完成 多 件 事 又 应 该 如 何 处 理 呢 ? 实际 上 ， 在 数据 库 中 也 经 常会 
遇 到 这 些 情 况 ,， 比 如: 多 个 用 户 同时 提交 数据 , 那么 , 在 数据 库 中 又 会 是 谁 提交 的 数据 呢 ? 
这 些 问 题 就 要 通过 事务 来 解决 了 。 
11.5.1 什么 是 事务 
事务 在 数据 库 中 的 地 位 就 像 交通 信号 灯 一 样 ， 信 号 灯 对 所 有 的 机 动车 和 非 机 动 都 很 重 
要 ， 而 事务 对 于 数据 来 说 也 是 非常 重要 的 。 如 果 能 够 合理 地 处 理事 务 ， 数 据 库 中 的 数据 就 
能 够 确保 安全 和 准确 了 。 那 么 ， 在 数据 库 中 究竟 什么 是 事务 呢 ? 非常 简单 ， 满 足下 面 4 个 
要 求 即 可 称 为 事务 。 或 者 说 是 事务 都 具有 下 面 这 4 个 特性 。 这 4 个 特性 就 是 ;原子 性 、 一 
致 性 、 隔 离 性 和 持久 性 。 下 面 就 让 我 们 与 它们 见面 吧 。 
口 原子 性 ， 也 称 为 事务 的 不 可 分 割 性 。 也 就 是 说 ， 在 数据 库 中 事务 中 的 每 一 部 分 都 
不 能 省 略 ， 不 能 只 执行 事务 中 的 一 小 部 分 ， 而 是 要 执行 事务 中 的 全 部 内 容 。 这 就 
好 像 是 洗衣 机 洗衣 服 ， 当 设 定好 一 个 执行 程序 ， 就 要 按照 这 个 程序 执行 ， 否 则 就 
会 完成 任务 。 
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口 一 致 性 : 是 指 事务 要 确保 数据 的 一 致 性 。 一 致 性 通常 是 指 不 论 数 据 如 何 更 改 都 要 
满足 数据 库 中 之 前 设置 好 的 约束 。 这 就 好 像 设置 了 旅程 的 起 点 和 终点 ， 无 论 如 何 
行走 ， 都 要 从 起 点 开始 ， 到 终点 结束 。 

口 隔离 性 : 是 指 每 个 事务 之 间 ， 在 执行 时 是 不 能 够 查看 中 间 状 态 的 。 也 就 说 事务 只 
有 提交 了 ， 才 能 够 看 到 结果 。 

口 持久 性 : 是 指 在 当 一 个 事务 提交 完成 后 ， 无 论 结果 是 否 正确 ， 都 会 将 结果 永久 保 
存在 数据 库 中 。 提 交 过 的 事务 是 不 能 够 恢复 的 。 


11.5.2 ”启动 和 保存 事务 

启动 事务 和 保存 事务 是 接触 事务 第 一 件 要 做 的 事情 ， 执 行 每 一 个 事务 时 都 要 先 告诉 数 
据 库 ， 现 在 要 开启 一 个 事务 ， 并 且 在 事务 执行 过 程 中 也 要 注意 设置 保存 点 ， 这 样 能 够 避免 
事务 出 现 错误 。 下 面 就 来 学 习 如 何 启动 和 保存 事务 。 

1. 启动 事务 

启动 事务 使 用 BEGIN TRANSACTION 语句 来 完成 。 具 体 的 语法 形式 如 下 


BEGIN { TRAN | TRANSACTION } transaction name 


这 里 ，transaction_name 为 事务 名 称 。TRAN | TRANSACTION 都 表示 事务 ， 用 哪个 都 
可 以 。 

2. 保存 事务 

保存 事务 与 保存 文件 有 些 类 似 ， 当 在 Word 中 写 东 西 的 时 候 ， 免 不 了 的 就 是 保存 。 有 
的 时 候 写 几 行 就 需要 保存 ， 当 后 面 的 内 容 写 错 了 ， 还 能 恢复 到 之 前 保存 的 状态 。 数 据 库 中 
的 事务 也 是 一 样 的 ， 可 以 通过 设置 保存 点 ， 来 保存 语句 执行 的 状态 ， 当 后 面 的 内 容 执行 错 
了 ， 还 能 够 回 滚 到 保存 点 。 保 存 事务 的 语法 形式 如 下 : 

SAVE { TRAN | TRANSACTION } savepoint name 

这 里 ，savepoint_name 是 保存 点 的 名 称 。 需 要 特别 注意 的 是 保存 点 的 名 字 和 变量 名 不 
同 ， 它 在 一 个 事务 中 是 可 以 重复 的 ， 但 是 ， 不 建议 读者 在 一 个 事务 中 设置 相同 的 保存 点 。 
如 果 设 置 了 重复 的 保存 点 ， 当 事务 需要 回 深 时 ， 只 能 回 深 到 离 当前 语句 最 近 的 保存 点 处 。 
11.5.3 ”提交 和 回 滚 事务 

有 了 事务 的 开启 和 设置 事务 的 保存 点 ， 接 下 来 就 是 最 关键 的 提交 事务 和 回 深 事 务 环节 
了 。 没 有 了 这 两 个 环节 ， 设 置 再 多 的 保存 点 也 没 办 法 完成 事务 的 操作 。 下 面 就 分 别 来 讲解 
如 何 提 交 和 回 深 事 务 。 

1. 提交 事务 

所 谓 提交 事务 ， 是 指 事务 中 所 有 内 容 都 执行 完成 。 这 就 好 像 是 考试 交卷 一 样 ， 如 果 提 
交 了 ， 就 不 能 再 进行 更 改 。 这 也 体现 了 事务 的 持久 性 的 特点 。 提 交 事 务 的 语法 形式 如 下 : 


COMMIT { TRAN | TRANSACTION } transaction name; 
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这 里 ，transaction name 是 指 事务 的 名 称 。 
2. 回 滚 事务 


回 滚 事务 就 是 可 以 将 事务 全 部 撤销 或 者 回 滚 到 事务 中 已 经 设置 的 保存 点 处 。 提 交 后 的 
事务 是 无 法 再 进行 回 深 的 。 
使 用 ROLLBACK TRANSACTION 回 滚 事 务 的 语法 结构 如 下 : 


ROLLBRACK {TRAN | TRANSACTION} 
[transaction name| savepoint name] 


| 


其 中 : 
口 transaction name: 事务 名 称 。 
口 savepoint name 项 : 保存 点 的 名 称 ， 必 须 是 在 事务 中 已 经 设置 过 的 保存 点 。 


11.5.4 ”事务 的 应 用 


读者 学 习 了 前 3 个 小 节 的 知识 后 , 相信 已 经 想 看 看 事务 是 如 何在 工 SQL 语句 中 应 用 的 
了 。 下 面 就 分 别 列举 一 个 提交 事务 的 示例 和 一 个 回 滚 事务 的 示例 来 演示 事务 的 使 用 方法 。 

【示例 13】 使 用 事务 完成 向 商品 信息 表 中 添加 1 条 数据 ， 并 提交 该 事务 。 

根据 题目 要 求 ， 具 体 的 语句 如 下 : 


USE chapterll; 


BEGIN TRANSACTION; -- 开 始 事务 
INSERT INTO productinfo VALUES (4, ' 靠 垫 ', 80, ' 车 饰 ', ' 北 京 ', '010-12348765'); 
COMMIT TRANSACTION; =-- 提 交 事 务 


执行 上 面 的 语句 ， 效 果 如 图 11.13 所 示 。 


< Mic oft SQL Server Management Studio -六 
文件 (E) 编辑 (E) ”视图 (W ”查询 (9Q) 项 目 (p) 调试 D) 工具 (DD 窗口 (WwW) 社区 (QO 和 帮助 (H) 
卫 新建 查询 | 让 也 也 玉 | 记 | 蕊 加 要 | 有 志 

My Wy? | chapterll -| ?1 执 和 5 》 和 v 品 晤 国 | 玉 杷 | 多 关 术 | 三 全 | 认证 


SQLQuery1.sql .istrator (52))* | 
vsE chapteril; 


BEGIN TRANSACTION; ””-- 开 始 事务 


INSERT INTO productinfo VALUES (4,' 靠 垫 ' ,80,' 车 饰 ',' 北 京 ','010-123468765'); 
COMMIT TRANSACTION; ”-- 查 交 事务 
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辐 消 息 | 


人 行 受 影响 ) 加 
J 四 


回 查询 已 成 功 执行 。 | WIDW-9527\MSSQLSERVER2008 (… | WIDW-9527\Administrato.,. | chapterll | 00:00:00 | 0 行 
就 绪 行 7 列 1 hl Ins 4 


图 11.13 ”提交 事务 的 应 用 


从 图 11.13 可 以 看 出 ， 通 过 提交 事务 的 语句 就 将 数据 添加 到 商品 信息 表 (productinfo) 
中 了 。 

【示例 14】 使 用 事务 完成 修改 商品 信息 表 中 编号 是 1 的 商品 价格 ， 将 其 价格 修改 成 
100， 并 给 其 操作 设置 一 个 保存 点 savepoint1; 然后 ， 再 删除 编号 是 1 的 商品 信息 ， 最 后 将 
事务 回 深 到 保存 点 savepointl 。 


ss 
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根据 题目 要 求 ， 具 体 的 语句 如 下 : 


USE chapterll; 


BEGIN TRANSACTION; -开始 事务 
UPDATE productinfo SET price=100 WHERE id=1; 
SAVE TRANSACTION savepoint17 -设置 保存 点 
DELETE productinfo WHERE id=1; 
ROLLBACK TRANSACTION savepoint1; -提交 事务 
执行 上 面 的 语句 ， 效 果 如 图 11.14 所 示 。 

古训 所 | 


文件 (E) ”编辑 (E) ” 视 苞 (W) ”查询 (9) 项 目 B) 调试 D) 工具 (D 窗 D(W 社区 (O 
帮助 (H) 


也 新建 查询 W) 六 他 渤 加 | 刘 | 芒 回 寺 癌 


a Wud | dapterll -| ?15 》 mv 史 铝 国 | 下 榴 | 镁 久 

SOL Ouery sa istrator CGI Ee 

日 USE chapterll; 下 
BEGIN TRANSACTION; -- 开 始 事务 


UPDATE productinfo SET price=100 WHERE id=1; 
SAVE TRANSACTION savepoint1; -- 设 置 保存 点 
DELETE productinfo WHERE id=1; 

ROLLBACK TRANSACTION savepoincl;  -- 提 交 事 务 


@ | wipw-ssz7WmssQLsERvER2008 (… | WIDW-9527\Administrato..，| chapterll | 00:00:00 | 0 行 
就 绪 行 9 列 ! Chi 


图 11.14” 回 深 事 务 的 应 用 


从 图 11.14 中 还 看 不 出 数据 表 的 具体 变化 ， 下 面 就 来 查看 商品 信息 表 中 id 为 1 的 记录 
是 否 存在 。 如 果 存 在 ， 就 说 明 事 务 回 滚 成 功 了 。 查 询 效果 如 图 11.15 所 示 。 


[twooson Sat Server Management Studio = 上》》 1 
文件 日 ”编辑 (6 视图。 查询 9) 项 目 (PB) 调 坛 D) 工具 (D 
窗 QOW， 社区 人。 帮助 


卫 章 寻 查询 由 | 记 | 孔 也 加 | 记 | 克 加 局 | 又 
By a | chapterll “| 了 执行 0 中 v 83 


SQLQuery1.sql -istrator (52))* -x 
SELECT * FROM productinfo; 一 
| 一 


北京 010.12345678 
沈阳 024-12345678 
上 海 021-12345678 
0-12348765 
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图 11.15 查询 商品 信息 表 中 的 数据 


从 图 11.15 所 示 的 界面 中 ， 可 以 看 到 id 为 1 的 记录 仍然 是 存在 的 ， 因 此 ， 也 就 是 说 
ROLLBACK 回 滚 操作 ， 确 实 是 将 语句 回 滚 到 了 保存 点 。 读 者 可 以 思考 一 下 ， 如 果 不 指定 
回 滚 的 保存 点 ， 会 出 现 什么 样 的 结果 呢 ? 
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11.6 本 章 小 结 


通过 本 章 的 学 习 ， 读 者 能 够 掌握 工 SQL 语句 的 基本 语法 规则 、 游 标 以 及 事务 的 使 用 。 
在 TSQL 语句 的 语法 规则 部 分 着 重 讲解 了 常量 、 变 量 以 及 流程 控制 语句 的 使 用 方法 ; 在 游 
标 部 分 ， 主 要 讲解 了 如 何 定 义 、 打 开 以 及 读 取 、 关 闭 游标 的 操作 ， 在 事务 部 分 ， 主 要 讲解 
了 事务 的 启动 、 保 存 点 设置 、 提 交 以 及 回 滚 事务 的 操作 。 


11.7 本 章 习题 


一 、 填 空 题 
1. 在 T-SQL 中 常量 的 前 级 是 
2. 循环 控制 语句 有 . 
3. 捕获 异常 的 语句 是 8 
二 、 选 择 题 
1. 下 列 哪 一 个 是 定义 游标 的 语句 
A. DECLARE cursor name B. CREATE cursor 
C. 以 上 都 不 对 
2. 下 面 哪 一 个 是 读 取 游标 的 语句 。 
A. read cursor B. use cursor C. fetch D. 以 上 都 不 对 
3. 下 面 对 事 务 的 描述 正确 的 是 5 
A. 提交 过 的 事务 还 可 以 回 滚 
B. 可 以 将 事务 回 滚 到 某 一 个 保存 点 
C. 只 能 将 事务 全 部 回 滚 
D. 以 上 都 不 对 


三 、 问 答题 

1. 事务 的 特点 是 什么 ? 为 什么 要 使 用 事务 ? 
2. 游标 使 用 的 4 个 步骤 是 什么 ? 

3， 如 何在 TSQL 中 获取 异常 。 

四 、 操 作 题 


更 改 示例 12 中 创建 的 游标 db_cursor2， 让 其 游标 查询 商品 价格 大 于 10 元 的 商品 ， 并 
使 用 FETCH 读 取 数 据 。 


.241 。 


第 12 章 一 次 编译 ， 多 次 执行 的 存储 过 程 


存储 过 程 是 TSQL 语句 主要 应 用 的 对 象 之 一 ， 在 存储 过 程 中 可 以 将 一 系列 相关 联 的 
SQL 语句 集合 到 一 起 。 如 果 想 执行 这 些 SQL 语句 ， 只 需要 通过 存储 过 程 的 名 字 就 可 以 调 
1 了 而 不 用 每 次 都 写 那 么 多 的 语句 。 既 然 存 储 过 程 给 用 户 带 来 了 如 此 大 的 好 处 ， 读 者 是 不 
是 应 该 集中 精力 把 它 学 好 呢 ? 

本 章 的 主要 知识 点 如 下 : 

口 认识 存储 过 程 

口 如 何 创建 存储 过 程 

口 如 何 修改 和 删除 存储 过 程 

口 如 何在 企业 管理 器 中 使 用 存储 过 程 


12.1 存储 过 程 很 强大 


存储 过 程 之 所 以 很 强大 不 仅 体现 在 它 在 执行 时 的 便利 性 ， 更 多 的 是 体现 了 它 的 安全 性 
和 可 重用 性 。 在 SQL Server 中 ， 存 储 过 程 不 仅 可 以 由 用 户 自 定义 ， 同 时 系统 中 也 提供 了 一 
些 直接 可 以 使 用 的 系统 存储 过 程 ， 方 便 用 户 使 用 。 在 本 节 中 将 带领 读者 认识 存储 过 程 的 
强大 。 


12.1.1 存储 过 程 的 特点 


存储 过 程 几乎 在 每 一 个 大 、 中 型 的 软件 系统 中 的 数据 库 设 计 中 存在 。 那 么 ， 为 什么 这 
些 软件 系统 的 数据 库 中 要 使 用 存储 过 程 呢 ? 如 果 您 还 不 清楚 原因 ， 就 请 看 下 面 的 存储 过 程 
的 特点 吧 。 

口 安全 性 。 存 储 过 程 之 所 以 安全 ， 是 因为 把 要 执行 的 SQL 语句 全 部 都 写 在 了 存储 过 

程 中 ， 而 在 程序 中 只 需要 通过 存储 过 程 名 来 调用 。 这 样 ， 就 有 效 保护 了 数据 库 中 
的 表 名 和 字段 名 ， 在 一 定 程度 上 提高 了 数据 库 的 安全 性 。 

口 提高 SQL 执行 的 速度 。 传 统 的 执行 SQL 语句 的 方法 , 每 次 执行 时 都 需要 对 语句 进 
行 编译 ， 然 后 再 执行 。 而 使 用 存储 过 程 ， 在 创建 存储 过 程 后 ， 只 要 执行 一 次 ， 以 
后 就 不 再 需要 编译 了 。 因 此 ， 存 储 过 程 被 称 为 是 一 次 编译 、 多 次 使 用 的 对 象 。 基 
于 存储 过 程 的 这 种 执行 方式 ， 就 大 大 提高 了 SQL 执行 的 速度 。 

口 提高 重用 性 。 所 谓 重 用 性 是 指 如 果 不 同 的 数据 库 有 者 相同 功能 的 需求 ， 那 么 ， 就 
可 以 直接 将 相应 的 存储 过 程 复制 过 去 ， 更 改 其 中 的 一 些 表 名 或 字段 名 就 可 以 了 。 

口 减少 服务 器 的 负担 。 服 务 器 每 天 要 执行 成 百 上 千 条 SQL 语句 。 如 果 能 将 一 些 数量 

比较 多 的 SQL 语句 都 写 入 存储 过 程 , 那么 , 在 执行 时 就 能 够 降低 服务 器 的 使 用 率 ， 
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同时 也 提高 了 数据 库 的 访问 速度 。 
12.1.2 ”存储 过 程 的 类 型 


在 SQL Server 中 存储 过 程 主 要 分 为 自 定义 存储 过 程 、 扩 展 存储 过 程 和 系统 存储 过 程 。 
其 中 ， 系 统 存储 过 程 实际 上 读者 已 经 不 陌生 了 ， 比 如 : 给 视图 改名 时 ， 用 到 的 系统 存储 过 
程 SP_RENAME; 扩展 存储 过 程 是 通过 编程 语句 创建 的 外 部 程序 ， 自 定义 存储 过 程 是 本 章 
要 学 习 的 主要 内 容 ， 就 是 通过 T-SQL 来 编写 的 实现 某 一 个 具体 功能 的 语句 集合 。 下 面 就 详 
细 介 绍 这 3 种 类 型 的 存储 过 程 。 


1. 自 定 义 存储 过 程 

自 定义 存储 过 程 是 在 数据 库 设 计 中 应 用 比较 多 的 。 在 自 定义 存储 过 程 中 可 以 传递 参 
数 ， 并 且 可 以 通过 存储 过 程 返回 参数 的 值 。 此 外 ， 自 定义 的 存储 过 程 还 可 以 通过 企业 管理 
器 来 创建 和 管理 。 


2. 扩展 存储 过 程 

扩展 存储 过 程 是 初学 者 用 得 比较 少 的 一 种 类 型 了 。 它 通常 是 使 用 C# 语 言 或 者 是 Java 
语言 来 编写 的 。 当 需要 使 用 时 直接 加 载 DLL 动态 链接 程序 就 可 以 了 。 

3. 系统 存储 过 程 

系统 存储 过 程 是 在 安装 数据 库 之 后 系统 自 带 的 存储 过 程 ， 它 可 以 直接 通过 存储 过 程 名 
来 调用 。 系 统 存储 过 程 是 可 以 给 用 户 使 用 数据 库 带 来 一 定 方便 的 ， 如 果 不 了 解 系统 存储 过 
程 都 有 哪些 ， 可 以 参考 SQL Server 的 帮助 文档 。 系 统 存储 过 程 除了 直接 调用 外 ， 还 有 一 个 
重要 的 特点 就 是 它 是 以 sp 为 前 绥 的 存储 过 程 。 


12.2 ”创建 存储 过 程 


了 解 了 存储 过 程 的 特点 后 ， 就 要 开始 步 入 存储 过 程 的 学 习 了 。 学 习 存 储 过 程 的 第 一 个 
环节 就 是 创建 存储 过 程 。 需 要 特别 提醒 读者 的 是 ， 在 本 章 中 学 习 的 都 是 自 定 义 存储 过 程 。 
自 定义 存储 过 程 既 可 通过 SQL 语句 创建 也 可 以 通过 企业 管理 器 来 创建 。 在 本 节 中 先 讲解 如 
何 通过 SQL 语句 来 创建 存储 过 程 , 在 本 章 最 后 一 节 中 将 统一 讲解 如 何在 企业 管理 器 中 创建 
和 使 用 存储 过 程 。 


12.2.1 创建 存储 过 程 的 语法 
存储 过 程 是 使 用 CREATE 语句 来 创建 的 ， 具 体 的 语法 形式 如 下 : 


CREATE { PROC | PROCEDURE } [schema name.] procedure name 
[ { @parameter data type } 
[VARYING] [= default] [[OUTPUT] 


el 
[ WITH <procedure option> [ ,...n] 
[ FOR REPLICATION ] 
AS { <sql statement> [;][ ...n]】} 
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[72] 
<procedure option> ::= 
[ ENCRYPTION ] 
[ RECOMPILE ] 
<sql statement> : := 
{ [ BEGIN ] statements [ END ] } 


其 中 : 
口 schema_name: 所 属 架 构 的 名 称 。 比 如 : dbo。 


口 procedure_name: 指 存储 过 程 的 名 称 。 由 于 系统 存储 过 程 的 前 级 是 SP_， 因 此 ， 在 


定义 存储 过 程 时 ， 不 要 以 SP 开头 。 
@parameter: 存储 过 程 中 的 参数 。 


default: 参数 的 默认 值 。 

OUTPUT: 指示 参数 是 输出 参数 。 
ENCRYPTION: 将 原始 文本 转换 为 加 密 格式 。 
RECOMPILE: 表示 存储 过 程 在 运行 时 编译 。 
-条 或 多 条 工 SQL 语句 。 


12.2.2 ”创建 不 带 参数 的 存储 过 程 


DOOOOOODO 


<sql_statement>: 


[type_schema_name.]data_type: 参数 以 及 所 属 架 构 的 数据 类 型 。 
VARYING: 指定 作为 输出 参数 支持 的 结果 集 。 仅 适用 于 游标 类 型 的 参数 。 


最 简单 的 一 种 自 定义 存储 过 程 就 是 不 带 参数 的 存储 过 程 ， 在 本 章 演练 存储 过 程 之 前 ， 


先 创建 数据 库 chapter12， 然 后 创建 公务 器 
(positioninfo )， 表 结构 如 表 12-1 和 表 12-2 所 示 。 


表 12-1 


公务 员 考 试 报名 信息 表 (reginfo) 


varchar(20) 


考试 报名 信息 表 (reginfo ) 和 职位 信息 表 


报名 人 


int 


年 龄 


varchar(5) 


性 别 


varchar(25) 


varchar(20) 


身份 证 号 
联系 方式 


varchar(20) 


家 庭 住 址 
报考 职位 编号 


数据 类 型 


说 


int 


编号 


varchar(20) 


这 里 ， 只 是 为 了 后 续 的 存储 过 程 编写 创建 方便 条 件 ， 实 际 上 ， 真 实 的 公 


信息 表 中 还 应 该 包括 更 多 的 信息 。 
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职位 名 称 


六 
次 
河 
dk 
喜 
荡 
RR 
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有 了 数据 表 ， 还 要 向 数据 表 中 填 入 数据 ， 填 入 的 数据 如 表 12-3 和 表 12-4 所 示 。 
表 12-3 ”公务 员 考 试 报名 信息 表 数 据 


联系 方式 


130123456789012 | 23467812 
12345678 
231234567890012 | 12345678 
12345678 
421123456789003 | 12345678 


职位 编号 职位 名 称 职位 名 称 
1 行政 执法 员 办 公 室 会 计 
2 审计 员 科普 知识 宣传 员 
3 资料 管理 员 


有 了 表 和 数据 的 准备 ， 下 面 就 使 用 示例 1 来 演示 如 何 创建 一 个 不 带 参数 的 存储 过 程 ， 
同时 也 熟悉 存储 过 程 创 建 的 语法 。 

【示例 1】 创建 存储 过 程 pro_1， 查 询 报 考 信 息 中 的 报考 人 、 性 别 以 及 报考 职位 名 称 。 

根据 题目 要 求 ， 该 存储 过 程 中 的 SQL 语句 只 是 一 条 查询 语句 。 具 体 的 语句 如 下 : 

CREATE PROCEDURE pro_1 

en name, sex, positionname FROM reginfo, positioninfo 

WHERE reginfo.positionid= positioninfo.id; 


执行 上 面 的 语句 ， 效 果 如 图 12.1 所 示 。 
[vcrosont SQL Server Management Studio 站 


文件 (E) ”编辑 () 视图 (yy 查询 (o) 项目 (6) ”调式 (0) 工具 (D ”窗口 (Ww) 社区 (O 
禾 助 (H) 
卫 新 建 查询 由 出 | 也 也 四 记 世 加 林带 瑟 
a Wa | chapter!2 -|? 扫 50 有 v 引 邓 [ 国 | 3 动 | 锥 和 
SUL OueryT sql istrator C3))*| 
日 CREATE PROCEDURE pro_ 1 

AS 


SELECT name,sex, positionname 了 RON reginfo, positioninfo 
WHERE reginfo. positionid= positioninfo.id 


性 滑 息 | 
命令 已 成 功 完成 。 引 


yl 只 


@ 到 | wIDW-9527\M55QLSERVER2008 (… | WIDW-9527\Administrato..，| chapter12 | 00:00:00 | 0 行 
就 绪 行 7 列 1 Chi ms /4 


图 12.1 创建 存储 过 程 pro_1 
至 此 ， 存 储 过 程 pro_1 已 经 创建 成 功 了 。 那 么 ， 如 何 查 看 存储 过 程 的 运行 效果 呢 ? 读 
者 可 以 回忆 之 前 是 如 何 调用 系统 存储 过 程 的 呢 ? 没 错 , 是 直接 用 存储 过 程 名 调用 的 。 但 是 ， 
如 果 是 自 定 义 的 存储 过 程 ， 就 要 使 用 EXECUTE 或 者 EXEC 来 调用 。 下 面 就 来 执行 存储 过 
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程 pro_ 1， 语 句 如 下 : 
EXEC pro_1; 
执行 上 面 的 语句 ， 效 果 如 图 12.2 所 示 。 


从 图 12.2 所 示 执行 存储 过 程 的 效果 可 以 看 出 ， 确 实 
是 符合 存储 过 程 中 查询 语句 的 要 求 。 


12.2.3 创建 带 输入 参数 的 存储 过 程 


在 上 一 小 节 中 已 经 学 习 了 不 带 参数 的 存储 过 程 的 使 
j， 实 际 上 ， 在 目前 的 数据 库 设计 中 带 参数 的 存储 过 程 
使 用 是 比较 多 的 。 存 储 过 程 中 的 参数 类 型 分 为 输入 参数 
和 输出 参数 。 下 面 先 以 示例 2 为 例 学 习 如 何 创建 带 输入 
参数 的 存储 过 程 。 

【示例 2】 创建 存储 过 程 pro_ 2， 根 据 输 入 的 报考 人 
姓名 ， 查 询 出 报考 人 的 年 龄 和 报考 职位 信息 。 


根据 题目 要 求 ， 报 考 人 可 以 作为 存储 过 程 的 输入 参数 。 


CREATE PROCEDURE Pro 2 name varchar (20) 
RS 
BEGIN 


SIx 
文件 (ED 编辑 @ 视图 查询 (Q) 项目) 
调 碟 0 工具 (D 窗口 QW 社区 (CO 天 助 (中 


明寺 WY | 让 | 人 他 咏 | 证 | 区 


pd 
2 | chapter12 = 
SQLQuery1.sql .-istrator Bis ba 
ExEcC pro_1; 
| 
到 行政 执法 员 


审计 员 

资料 管理 员 
科普 知识 宣传 员 
科普 知识 宣传 员 


bw-s527Wdministrato,，| chapterl2 | 00:00:00 | 5 行 
就 绪 行列 1 ch! 才 


图 12.2 执行 存储 过 程 pro_1 
创建 语句 如 下 : 


SELECT age,positionname FROM reginfo,positioninfo 
WHERE reginfo.positionid=positioninfo.id AND reginfo.name=@name; 


END 


执行 上 面 的 语句 ， 效 果 如 图 12.3 所 示 。 


文件 (E) ”编辑 (E) ”视图 (W) 查询 (9) 项目 (P) 调试 D) 工具 (D) 窗口 (W) ”社区 (CO) ”和 必 助 (H) 


BL MEE 


=I9|x| 


By 2 | chapter12 


“| ?执行 0 下 v ?3 可 [ 国 | 3? 艳 | 引力 | 办 


SQLQuery1.sql -istrator (53))* | 


| 
x 


章 CREATE PROCEDURE pro 2 
AS 
BEGIN 

= SELECT age,positionname FRONM reginfo,positioninfo 


Bname varchar (20) 


END 


WHERE reginfo.positionid=positioninfo. id AND reginfo.name=Bname; 


. 昌 


己 消 息 | 
命令 已 成 功 完成 。 
加 
多 查询 已 成 | wIDW-9527WM55QLSERYER2008 (… | WIDW-9527WAdministrato… 


| chapter12 | 00:00:00 | 0 行 


上 


就 绪 行 3 列 !1 
图 12.3 ”创建 存储 过 程 pro_2 


hl Ins 了 


执行 带 输入 参数 的 存储 过 程 仍然 使 用 的 是 EXECUTE/EXEC 关键 字 ， 但 是 执行 带 参 数 
的 存储 过 程 还 要 注意 传递 参数 ， 并 且 参 数 的 类 型 也 要 匹配 。 执 行 存储 过 程 pro_ 2 的 语句 


如 下 : 


EXEC Pro_2 “王丽丽 " 
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执行 上 面 的 语句 ， 效 果 如 图 12.4 所 示 。 


全 注意 : 在 调用 带 参数 的 存储 过 程 时 ， 传 递 参 数 的 个 数 和 数 。 机 四 IO 全 
据 类 型 一 定 要 与 调用 的 存储 过 程 相 匹配 。 此 外 ， 在 传 ” 沁 旨 二 au| 记 | 也 也 怠 | 马 加 


递 日 期 时 间 类 型 和 字符 囊 类 型 的 数据 时 ， 还 要 注意 给 时 基 he 

这 些 数据 加 上 单 引号 。 和 
| ml 
器 结果 | .2 消息 | 


12.2.4 创建 带 输出 参数 的 存储 过 程 


存储 过 程 中 的 默认 参数 类 型 是 输入 参数 ， 如 果 要 为 存储 。 到 Wannaraa |depena mooon | 1 生 
过 程 指定 输出 参数 , 还 要 在 参数 类 型 后 面 加 上 OUTPUT 关键 
字 。 下 面 就 使 用 示例 3 演示 如 何 创建 带 输出 参数 的 存储 过 程 。 


ol 
文件 (E) ”编辑 (E) ”视图 (W) 查 淘 (Q) 


社区 (CO 天 助 (H) 


就 绪 1h: / 
图 12.4 执行 存储 过 程 pro_2 


【示例 3】 创建 存储 过 程 pro 3， 根据 输入 的 报名 号 ， 输 出 该 考生 的 姓名 和 年 龄 。 
根据 题目 要 求 ， 在 存储 过 程 pro_3 中 共有 3 个 参数 ， 包 括 1 个 输入 参数 和 2 个 输出 参 


深 


创建 语句 如 下 : 


CREATE PROCEDURE pro 3 


@id int,@name varchar (20) output,@age int output 


AS 
BEGIN 


SELECT name=name @age=age FROM reginfo 


WHERE id=@id; 
END 


执行 上 面 的 语句 ， 效 果 如 图 12.5 所 示 。 


执行 存储 过 程 pro 3， 输入 参数 的 值 需要 传递 ,但 是 输出 参数 的 值 是 不 需要 传递 的 。 


执行 语句 如 下 : 


DECLARE @x varchar (20), @y int 
EXEC pro 3 1201,@x output,@y output 
SELECT @x, @y; 


执行 上 面 的 语句 ， 效 果 如 图 12.6 所 示 。 


二 
文 作 虽 。 畏 句 ( 旬 。 视图 由 查询) 项 目 (E) 调式 D) 工具 ID 窗 D) 
社区 (C) 才 助 由 
了 新 于 查询 | 昼 信 谤 钙 | 虽 | 区 加 时 加 
Ry | chapter!2 -| 1 扫 和 WwW >》 v 中 昌国 | 
SQLQuery1.sql -istrator (53))* | -Xx 

日 CREATE PROCEDURE pro 3 
Bid int,Bname varchar (20) output,Rage int output 
as 

DBEGIN 

D SELECT Bname=namevBage=age FRON reginfo 

上 WHERE id=Bid; 

上 END 


VER2008 (... | WIDW-9527\Administrato...| chapterl2 | 00:00:00 | 0 行 


-- 声 明 输出 参数 名 
-- 显 示 输 出 参数 的 值 


aly 
文件 (E) ”编辑 (E) ”视图 (YW) 查询 (9) 项目 ) ”调式 (D) 

工具 (D 窗口 Ww) 社区 (帮助 中 

也 新 建 查询 | 让 轧 也 下 | 记 | 蕊 回忆 | 过 画 

上 3 | chapterl2 


SQLQuery1.sql -istrator (53))* 
日 DEFCLARE Bx varchar (20), By int 


EXEC pro 3 1201,Bx output,By output 
SELECT Bx, By; 


司 结果 | 己 消息 | 
[| Fs) | FS) | 
IT kh | 5 


ER2008(..，| WIDW-9527\Administrato,.，| chapterl2 | 00:00:00 | 1 行 


行 10 列 1 chi 


图 12.5 创建 存储 过 程 pro 3 


就 绪 行 5。 列 1 Chl 


图 12.6 执行 存储 过 程 pro 3 
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12.2.5 创建 带 加 密 选 项 的 存储 过 


所 谓 加 密 选项 并 不 是 对 存储 过 程 中 查询 出 来 的 内 容 加 密 ， 而 是 将 创建 存储 过 程 本 身 的 
语句 加 密 。 通 过 对 创建 存储 过 程 的 语句 加 密 ， 可 以 在 一 定 程度 上 保护 存储 过 程 中 用 到 的 表 
信息 ， 同 时 也 提高 了 数据 库 的 安全 性 。 下 面 就 使 用 示例 4 来 演示 如 何 创 建 带 加 密 选项 的 存 
储 过 程 。 

【示例 4】 创建 带 加 密 选项 的 存储 过 程 pro_4， 查 询 报考 人 的 姓名 、 年 龄 、 性 别 和 地 址 
信息 。 

根据 题目 要 求 ， 带 加 密 选项 的 存储 过 程 使 用 的 是 with encryption。 创 建 存储 过 程 的 语 
名 如下; 

CREATE PROCEDURE Pro_4 

WITH ENCRYPTION 

RS 

BEGIN 


SELECT name,age, sex,address FROM reginfo 
END 


执行 上 面 的 语句 ， 效 果 如 图 12.7 所 示 。 
执行 存储 过 程 pro_4 的 语句 如 下 : 


EXEC pro_4; 


执行 上 面 的 语句 ， 效 果 如 图 12.8 所 示 。 


丁丁 站 | 
文件 (E) ”编辑 (E) ”视图 (YV) ”查询 (Q) 
项 目 (P) 调试 O) 工具 (D 窗口 (WW) 
社区 (QO 必 助 (中 

也 和 本 | 记 | 一 也 本 | 记 中 
BY Ha | chapter12 


ecee 
SQLQuery1.sql .istrator (53))* | w= 
日 EXEC pro_4; 习 
-由 a 四 = 


北京 
王丽丽 21 女 大 连 
吴 笋 ”30 女 ”沈阳 
超 昌 |29 | 男 | 上海 
| | 其 % 网 “26 女 “南京 


有 、Microsoft SQL Server Management Studio =|lo|x| 
文件 日 ”编辑 (E) ”视图 (W 查询 (9) 项 目 (PB) 调试 D) 工具 

窗口 (WW) 社区 (QO 帮助 (H) 

也 .新建 查 W | 六 | 人 壕 信 咏 | 忆 | 让 回忆 | 癌 

| chapter12 | ?执行 0 和 VV 中 时 


SQLQuery1.sql -istrator (53))* | 
ECREATE PROCEDURE pro_4 
| ENCRYPTION 


SELECT name,age, sex,address FROM reginfo 
-END 


局 消息 | 

命令 已 成 功 完成 。 
避 2 
|55QLSERVER2008 (… | WIDW-9527\Administrato..，| chapter12 | 00:00:00 | 0 行 
就 绪 行 8 列 1 chl Ed 


图 12.7 创建 存储 过 程 pro_4 图 12.8 ”执行 存储 过 程 pro_4 
通过 图 12.8 所 示 执 行 存储 过 程 的 效果 ,读者 会 想 , 也 没 看 出 来 加 密 后 的 存储 过 程 有 什 
么 不 同 啊 。 在 创建 加 密 存储 过 程 之 前 ， 就 已 经 向 读者 说 明 过 了 ， 加 密 存 储 过 程 并 不 是 对 存 
储 过 程 的 结果 加 密 而 是 对 创建 语句 加 密 的 。 那 么 ， 如 何 查 看 存储 过 程 的 创建 语句 呢 ? 使 用 
系统 存储 过 程 SP_HELPTEXT 就 可 以 查看 了 。 查 看 存储 过 程 pro_4 的 语句 如 下 : 


SP_HELPTEXT pro 4; 


ladministrato.,,| chapter12 | 00:00:00 |5 行 
就 绪 Nt 4 
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执行 上 面 的 语 滞 句 ， 效 果 如 图 12.9 所 示 。 
通过 图 12.9 显示 的 结果 ， 就 可 以 看 出 pro_ 4 是 加 密 后 的 存储 过 程 ， 是 无 法 查看 到 创建 
语句 的 。 为 了 让 读者 对 系统 存储 过 程 SP_HELPTEXT 印象 深刻 ， 下 面 使 用 该 存储 过 程 查 看 
之 前 创建 的 pro_3， 语 句 如 下 : 


SP_HELPTEXT pro_3; 


执行 上 面 的 语句 ， 效 果 如 图 12.10 所 示 。 


=|Ox 
文件 [E) ”编辑 (E) 视图 W)， 查询 (Q) ”项目 E) 凋 式 D) 工具 ID 
窗口 W) 社区 (QO 帮助 办 


了 新建 查 汉 | 六 | 溃 久 号 | 让 | 苞 回 马 | 加 


ol 
文件 (E) ”编辑 (E) ”视图 (Y) ”查询 (Q) 


项 目 B) 调试 D) 工具 (D 窗口 (wD) 
社区 (和 灵 助 由 

卫 . 利 建 查 区 由 | 出 | 也 也 马 | 记 三 
村 ti | chapter12 
SQLQuerylsql -istrator (53))* 
Me :>p_helpreexc pro 3: 
已 消息 | 

对 象 pro_4， 的 文本 已 加 密 。 
| 


CREATE PROCEDURE pro_3 


Gd ntGname varchar(20] outpuLGage nt output 


,| WIDW-9527\Administrato,,，| chapter12 | 00:00:00 | 17 行 
行 ! 列 ! hl 


图 12.9 查看 pro_4 的 创建 语句 图 12.10 查看 pro_3 的 创建 语句 


从 查询 pro 3 和 pro_4 的 结果 可 以 看 出 ， 加 密 后 的 存储 过 程 确实 能 够 提高 数据 库 的 安 
全 性 。 


12.3 ”修改 存储 过 程 


存储 过 程 虽然 很 复杂 ， 但 是 在 创建 后 也 是 可 以 修改 的 。 在 修改 存储 过 程 前 ， 也 要 确保 
之 前 的 存储 过 程 中 的 内 容 不 再 使 用 了 ， 否 则 ， 修 改 之 后 也 是 不 能 够 恢复 的 了 。 实 际 上 ， 只 
要 读者 掌握 了 创建 存储 过 程 的 语法 ， 修 改 存储 过 程 可 以 说 是 小 菜 一 碟 。 


12.3.1 修改 存储 过 程 的 语法 


修改 存储 过 程 的 语法 与 创建 存储 过 程 的 语法 类 似 ， 只 不 过 是 将 CREATE 改 成 了 
ALTER。 有 具体 的 语法 形式 如 下 : 


ALTER { PROC | PROCEDURE } [schema name.] procedure name 
[ { eparameter data type } 
[VARYING] [= default] [[OUTPUT] 
而 we Rem) 
[ WITH <procedure option> [ ,...n] 
[ FOR REPLICATION ] 
B31{ <saql statement> [lf 0 
La 


<procedure option> ::= 
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[ ENCRYPTION ] 
[ RECOMPILE ] 
<sql statement> ::= 
{ [ BEGIN ] statements [ END ] } 


上 面 的 语法 形式 已 经 在 创建 存储 过 程 的 时 候 讲 解 过 了 ， 如 果 读者 还 有 疑问 ， 可 以 参考 
创建 存储 过 程 时 的 语法 解释 。 


12.3.2” 改 一 改 存储 过 程 


下 面 就 到 了 演练 修改 存储 过 程 的 环节 了 。 在 修改 存储 过 程 的 过 程 中 其 实 也 是 帮助 读者 
把 创建 存储 过 程 的 语句 再 熟悉 一 遍 。 下 面 就 通过 示例 5 和 示例 6 来 演示 如 何 修改 存储 过 程 。 

【示例 S】 修改 存储 过 程 pro_1， 只 查询 出 报考 人 和 报考 职位 信息 。 

根据 题目 要 求 ， 修 改 存储 过 程 pro_1 的 语句 如 下 : 

ALTER PROCEDURE pro_1 

RS 

SELECT name, positionname FROM reginfo, positioninfo 

WHERE reginfo.positionid= positioninfo.id; 

执行 上 面 的 语句 , 效果 如 图 12.11 所 示 。 修改 后 的 存储 过 程 pro_1, 执行 效果 如 图 12.12 
所 示 。 


[mcrosore at Server Monsoon Ie ET 
文件 (E) ”编辑 (E) ”视图 (Y) 查 淘 (Q) 


二 加 本 
文件 (E) ”编辑 (E) 视图 ( 几 。 查询 (9) 项目) 调试 D) 工具 (D 窗口 (Ww) oy om 本 
AE(O TM) 守山 售 汕 吕 | 咏 。 避 
也 ae | 六 | 钙 耳 久 | 忆 | 区 回馈 | 加 电 有 ee - 
By 2 | chapterl2 -| 执行 WW 》 及 咬 加 FH” SQLQuerylsql -istrator (53))* = 
SOL QuveryL.sql ~istrator (53))*| x | i — 


间 ALTER PROCEDURE pro_1 
| 25 
SELECT name, positionname FROM regi 
| wa reginfo.positionid= positioni 


资料 管理 员 
科普 知识 
5_ | 黄 珊 形 ”科普 知识 


已 消息 | 
命令 已 成 功 完成 。 


EL 
WN-9527\MSSQLSERYER2008 (.,，| WIDW-9527VAdministrato,..| chapter12 | 00:00:00 | 0 行 ps27\administrato, ,| chapter12 | 00:00:00 | 5 行 


就 绪 行 6 列 1 Chl 4 就 绪 生 h 和 
图 12.11 修改 存储 过 程 pro_1 图 12.12 执行 修改 后 的 存储 过 程 pro_1 


名 说明: 使 用 ALTER 语句 修改 存储 过 程 ， 不 能 够 修改 存储 过 程 的 名 字 。 


【示例 6】 修改 存储 过 程 pro 2， 将 其 修改 成 加 密 的 存储 过 程 。 
根据 题目 要 求 ， 修 改 pro_2 的 语句 如 下 : 


ALTER PROCEDURE Pro 2 name varchar (20) 

WITH ENCRYPTION 

RS 

BEGIN 

SELECT age,positionname FROM reginfo,positioninfo 

WHERE reginfo.positionid=positioninfo.id RND reginfo.name=@name; 
END 
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执行 上 面 的 语句 ， 效 果 如 图 12.13 所 示 。 
下 面 使 用 系统 存储 过 程 SP_HELPTEXT 查看 存储 过 程 pro_ 2 的 创建 语句 ， 验 证 是 否 已 
经 对 其 加 密 了 。 查 询 效 果 如 图 12.14 所 示 。 


从 图 12.14 的 效果 中 可 以 看 出 , 确实 是 将 pro_2 的 存储 过 程 修改 成 加 密 的 存储 过 


了 Microsoft SQL Server Management Studio 


x 
Er 编辑 (E) ED 可 局 Q) 项 上 调 或 D， 工具 D 定 D(W) iE(O WWD) 
卫 新 建 查 光 W | 让 | 孔 隅 加 | 让 | 区 号 二 | 怠 瑟 
Pe CA TE 让 EEC 
“SQLQueryl.sql -istrator (53))* | ”x 文件 (日 ”编辑 (E) ”视图 (W 查 漳 (9) 项目 (P) 
BALTER PROCEDURE pro 2 Bname varchar (20) 一 调 坛 0) 工具 ID) 窗口 (w) 社区 (C)” 必 助 (H) 
WITH ENCRYPTION 
|s es | 动 马 瑟 | 避 | 此 吕 昌 
BEGIN 
DSELECT age, positionname FRON reginfo,positioninfo 对 级 | chapterl2 4 
WHERE reginfo. positionid-positioninfo. id AND reginfo. name-Bname; SOL QueryL.sql istrator (53))* -x 
END 了 加 SP_HELPTEXT pro 2; 下 
1 » 
| 局 消息 | 
命令 已 成 功 完成。 ~ 对 象 pre_z' 的 文本 已 加 密 . 
可 se | 2 
加 查询 已 成 ，| wlIDW-9527\Ms5QLSERVER2008(..，| WIDW-9527\adiministrato..，| chapterl2 | 00:00:00 | 0 行 low-9527\dministrato,,，| chapter12 | 00:00:00 | 0 行 
就 绪 行 !1 列 ! chi ms /| 就 绪 行 员 1 chl 5 
图 12.13 ”修改 存储 过 程 pro_2 图 12.14 查看 pro_2 的 创建 文本 


12.3.3 ”给 存储 过 程 改 个 名 


读者 应 该 还 记得 如 何 修改 视图 的 名 称 吧 ， 没 错 ， 修 改 存储 过 程 也 是 通过 系统 存储 过 程 
SP_RENAME 来 完成 的 。 实 际 上 ， 在 设计 数据 库 时 就 把 存储 过 程 的 名 字 已 经 规定 好 了 ， 要 
尽量 少 修改 存储 过 程 的 名 字 ， 以 免 给 其 他 引用 存储 过 程 的 对 象 的 使 用 造成 错误 。 下 面 就 通 

过 示例 7 来 演示 如 何 给 存储 过 程 改 名 。 
【示例 7】 将 存储 过 程 pro_1 的 名 字 更 改 成 pro_new。 
根据 题目 要 求 ， 使 用 SP_RENAME 系统 存储 过 程 改 名 ， 语 句 如 下 : 


SP_RENAME pro 1, pro_new; 


执行 上 面 的 语句 ， 效 果 如 图 12.15 所 示 。 


< Microsoft SQL Server Management Studia =| 口 | x 
民 TTOETOETO 

家 Do 六 区 OO 帮助 由 

也 新 建 查 星 | 册 六 证 | 访 | 芒 回 对 | 册 s 

再 让 | chapterl2 “| ?执行 0 pV 号 加 
SQLQuery1.sql .istrator (53))* 

日 SP_RENANE pro_1, pro_nevw; 


a 

已 消息 | 

注意 : 更 改 对 象 名 的 任 一 部 分 部 可 能 会 破坏 本 和 存储 过 程 。 
| 


YER2008 (.,，| WIDW-9527WAdministrato.,，| chapterl2 | 00:00:00 | 0 行 
行 4 列 1 Chi 


图 12.15 使 用 SP_RENAME 改名 
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12.4 删除 存储 过 程 


当 存储 过 程 不 再 需要 时 ， 可 以 将 其 从 数据 库 中 删除 。 但 是 ， 在 删除 前 一 定 要 确保 存储 
过 程 没 有 被 其 他 对 象 所 使 用 ， 和 否则 就 会 出 现 错误 。 另 外 ， 删 除 后 的 存储 过 程 是 不 能 够 恢复 
的 ， 因 此 ， 删 除 前 请 读者 一 定 要 将 存储 过 程 查看 清楚 或 者 对 其 备份 。 


12.4.1 删除 存储 过 程 的 语法 


删除 存储 过 程 与 删除 其 他 数据 库 对 象 的 语法 类 似 ， 都 是 使 用 DROP 语句 来 删除 的 。 具 
体 的 语法 形式 如 下 : 


DROP PROC Pro_name, [..n]; 


这 里 ，pro_name 是 存储 过 程 的 名 字 。 在 删除 存储 过 程 时 ， 可 以 同时 删除 1 到 多 个 存储 
过 程 ， 多 个 存储 过 程 之 间 用 去 号 隔 开 就 可 以 了 。 


12.4.2 ”清理 不 用 的 存储 过 程 


按照 上 一 小 节 讲 解 的 删除 存储 过 程 的 语法 ， 在 本 小 节 中 将 通过 示例 8 和 示例 9 分 别 演 
示 如 何 删除 1 个 和 多 个 存储 过 程 。 

【示例 8】 先 创 建 存储 过 程 pro 5， 查询 职位 信息 后 再 将 其 删除 。 

根据 题目 要 求 ， 具 体 语句 如 下 : 


CREATE PROCEDURE Pro_5 

RS 

BEGIN 

SELECT positionname FROM positioninfo; 
END 

DROP PROC pro_5; 


执行 上 面 的 语句 ， 效 果 如 图 12.16 所 示 。 

【示例 9】 分 别 创建 两 个 存储 过 程 pro_5 和 pro 6， 全 是 查询 职位 信息 的 。 然 后 将 这 两 
个 存储 过 程 一 并 删除 。 

根据 题目 要 求 ， 具 体 语 句 如 下 : 


CREATE PROCEDURE pro_5 

AS 

BEGIN 

SELECT positionname FROM positioninfo; 
END 

GO 

CREATE PROCEDURE pro_6 

RS 

BEGIN 

SELECT positionname FROM positioninfo; 
END 

DROP PROC pro_ 5, pro_ 6; 
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执行 上 面 的 语句 ， 效 果 如 图 12.17 所 示 。 


a Microsoft SQL Sery 


er Management Studig 

文件 (E) ”编辑 (E) 视图 几 。 查询 Q) 项目 EB) 调试 D) 
工具 (D 窗口 W) 社区 (QO 帮助 (H) 

:及 间 当 放 六 | 售 六 态 | 启 | 芒 回 对 | 癌 忆 
MY Wa | chapter12 后 


SQLQuery1.sql -istrator (52))* 
日 CREATE PROCEDURE pro_5 


-次 编译 ， 


多 次 执行 的 存储 过 程 


A Microsoft SQL Server 


Management Studio 
文件 日 ”编辑 {E) ”视图 (VW) ”查询 (9) 项 目 (P) 调试 D) 工具 (D 
窗口 (WW) 社区 (QO 帮助 

巡 亲 机 WD 六 | 他 这 地 | 局 | 蕊 回 筷 | 台 5 

对 禄 | hapteriz "|W PV 各 
SQLQuery1.sql -istrator (52))* 
日 CREATE PROCEDURE pro_5 

| AS 

BEGIN 

| SELECT positionname FRON positioninfo; 
-END 


CREATE PROCEDURE pro_6 
I 
由 BEGIN 


| SELECT positionname FRO positioninfo; 


| En 
| PROC pro_5,pro_6; 


END 
上 DROP PROC pro_5: 


a5 
9 BEGIN 
SELECT positionneme 了 FROM positioninfo; 


008 (… | WIDW-9527\Administrato.,，| chapter12 | 00:00:00 | 0 行 
就 绪 行 8 列 ! chi 


,| chapterl2 | 00:00:00 | 0 行 
chl 


图 12.16 创建 并 删除 存储 过 程 pro_5 
好 了 ， 删 除 存储 过 程 的 方法 读者 应 该 明白 了 吧 。 


12.$ 使 用 企业 管理 器 管理 存储 过 


图 12.17 ”删除 多 个 存储 过 程 


通过 前 面 儿 节 的 学 习 ， 相 信 读 者 已 经 发 现 创建 存储 过 程 还 是 挺 麻烦 的 。 但 是 ， 有 了 企 
业 管理 器 直接 创建 和 管理 存储 过 程 就 能 方便 多 了 。 在 本 节 中 ， 就 将 讲述 如 何 使 用 企业 管理 
器 来 创建 、 修 改 以 及 删除 存储 过 程 。 有 了 方便 的 工具 ， 你 还 在 等 什么 呢 ? 


12.5.1 使 用 企业 管理 器 创建 存储 过 程 


使 用 企业 管理 器 创建 存储 过 程 ， 相 对 于 使 用 语句 创建 存储 过 程 是 容易 得 多 ， 至 少 不 用 
记 住 那么 多 的 关键 字 。 但 是 ， 在 创建 存储 过 程 时 ， 还 是 需要 很 多 步骤 的 ， 需 要 读者 记 清 楚 
哦 ! 下 面 就 通过 示例 10 来 演示 如 何在 企业 管理 器 中 创建 存储 过 程 。 

【示例 10】 使 用 企业 管理 器 创建 之 前 创建 过 的 存储 过 程 pro_3, 并 将 其 命名 为 pro 31。 
根据 输入 的 报名 号 ， 输 出 该 考生 的 姓名 和 年 龄 。 

在 企业 管理 器 中 创建 存储 过 程 ， 需 要 通过 以 下 5 个 步骤 来 完成 。 

(1) 在 “对 象 资源 管理 器 ”窗口 中 ， 展 开 chapter12 节点 。 

(2) 在 chapter12 的 节点 下 ， 找 到 “可 编程 性 ”节点 ， 并 将 其 展开 。 

(3) 在 “可 编程 性 ”节点 下 ， 右 击 “ 存 储 过 程 ” 节 点 ， 在 弹出 的 右键 菜单 中 选择 “新 
建 存储 过 程 ”选项 ， 打 开 新 建 存储 过 程 界面 ， 如 图 12.18 所 示 。 

(4) 在 图 12.18 所 示 界 面 中 给 出 了 创建 存储 过 程 的 基本 语法 框架 ， 只 需要 添加 相应 的 
参数 和 语句 就 可 以 完成 了 。 按 照 题目 要 求 ， 添 加 后 的 效果 如 图 12.19 所 示 。 
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ent Studio 
文才 日” 编 扣 6 视图 W)， 查 词 O) 项 目 B) 调 D) 工具 (D 宣 D(W) 社区 (O 帮助 
2 | 六 六 名 a 总 s 


本 ;下 | chapter!2 -| 执行》 和 v 弹 各国 |” 弘 | 六 央 宫 | 三 4 
“SQLQueryz.sql -nistrator (53)) SQLQusry1.5 .istrator (52)* | -Xx 
SET ANSI_NULLS ON = 
oo 


SET QUOTED_IDENTIFIER ON 


<Author, , Name> 
: <Create Date,,> 


日 CREATE PROCEDURE <Procedure_ Name, v edureName: 
-- Add the paramerera for che stored procedure here 


BParaml, sysname, Bpl> <Datatype For Paraml, , int> = <Default_ yl 
BPparam2, sysname, Bp2> <Datatype For Param?, , int> = <Default V 
5 
BEGIN 
加。 SET NocOUNT ON added to prevent extra result sets trom 


-- interfering vith SELECT statements. 
SET NOCOUNT ON; 


-- Insert statements for procedure here 
SELECT <Rparaml, ayanae, Rpl>, SAParam?, ayaname, Rp2> 


图 12.18 ”新 建 存储 过 程 界面 


文件 日 ”编辑 (E) ” 视 苞 W) 查询 (9) 项 目 (E) 调式 0) 工具 (D 窗口 (W) 社区 (QO 帮助 (中) 

;也 新 建 查询 由 | 让 | 这 也 加 由 | 蕊 加 要 | 过 天 

图 记 | oo -| 1 执 5W 》 mv 民生 加 站 弘 | 如 图 涪 | 三 | 来 这 | 包 E 
SQLQuery1.sql -istrator (52))* | 


Er | 村 对 上 了 FE 一 che definition of the procedure. 


SET ANSI_NULLS ON 
Go 
SET QUOTED_IDENTIFIER ON 


Auchor <hAuthor, ,Name> 
Create date: <Create Date,,> 


BCREATE PROCEDURE pro_31 
-- Add the parameters for the stored procedure here 
Bid intvBname varchar (20) output,Bage int output 
5 
BEGIN 
~- SET NOCOUNT ON added to prevent extra result sets from 
-- interfering with SELECT statements. 
SET NOCOUNT ON; 


-- Insert statements for procedure here 
SELECT Bname=name,Rage=age FRON reginfo WHERE id=Bid; 


» 
~ | WIDW-9527\MS5QLSERVER2008 (… | WIDW-9527VAdministrato..，| chapter12 | 00:00:00 | 0 行 
行 % 列 1 ahi Ins 用 


图 12.19 ”添加 参数 后 的 界面 


(5) 检查 图 12.19 界面 中 添加 后 的 语句 ， 如 果 没 有 问题 ， 就 可 以 通过 选择 菜单 “查询 ” 
|“ 执 行 ” 选 项 ， 来 执行 存储 过 程 创建 命令 ， 完 成 存储 过 程 的 创建 操作 。 如 图 12.20 所 示 。 

如 果 要 运行 存储 过 程 ， 也 可 以 通过 企业 管理 器 完成 。 在 “可 编程 性 ”|“ 存 
储 过 程 ” 节 点 下 ， 右 击 pro_31 节点 ， 在 弹出 的 右键 菜单 中 选择 “执行 存储 过 程 ” 选 项 ， 弹 
出 图 12.21 所 示 界 面 。 
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crosoft SQL Server Management Studio = | x| 
文件 中 编辑 视图 查询 0) 项 目 @) 调试 D) 工具 (7) 窗口 (W) 社区 (C) 下 助 (4 
:和 | 访 | 杞 杞 名 | 访 | 玉器 马 | 加 

: By Ha | chapter12 <|? 扫 Fo 有 v 品名 国 | 下 弹 | 痊 疾 名 | 三 


“SQLQuery1.sql .istrator (52))* vx 
日 CREATE PROCEDURE pro_31 
-- Add the parameters for the stored procedure here 
Bid intvBname varchar (20) output,Bage int output 


ES 

BEGIN 
-- SET NOCOUNT ON added to prevent extra result sets from 
-- interfering with SELECT statements. 
SET NOCOUNT ON; 


-- Insert statements for procedure here 
SELECT Bname=name,Bage=age FRON reginfo WHERE id=Bid; 


WIDW-9527\MS5QLSERVER2008 (..，| WIDW-9527\Administrato..，| chapter12 | 00:00:00 | 0 行 


"执行 过 程 


连接 


吊 查看 连接 属性 


进度 


服务 器 
WIDN-9527 \MSSQLSERVER2008 


连接 
MTDY-9527\Adninistrator 


行 38 列 ! Chl Ins 


图 12.20 ”执行 存储 过 程 创建 命令 


][pro_31] 


-wy 


在 图 12.21 所 示 的 界面 中 ， 可 以 看 到 在 存储 过 程 中 设置 的 3 个 参数 ， 在 这 是 


图 12.21 运行 存储 过 程 添加 参数 界面 


已， 输入 相 


应 的 参数 值 即 可 。 由 于 在 这 3 个 参数 中 ， 只 有 @id 是 输入 参数 ， 因 此 ， 只 需要 给 @id 赋值 。 
这 里 ， 给 其 赋值 1201。 单 击 “确定 ”按钮 ， 运 行 效果 如 图 12.22 所 示 。 
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x Microsoft SQL Server Management Studio 


文件 四 ”编辑 EB) 视 攻 (WW) ”查询 (Q) 项 目 P) 调式 D) 工具 (D 窗口 (Ww 


社区 (OO PW) 
;了 ae | 入 | 钙 孔 名 | 广 | 蕊 日 马 | 如 局 
:By BR | dapterlz -|? 扫 5 》 加 v 吧 富国 各 


SQL Query2.sql -nistrator (54))! SQLQueryl.sq .istrator (2) | vx 
USE [chapter12] 
60 


日 DECLARE Brecurn value int, 
Bname varchar (20), 
Bage int 


EXEC ~ Breturn value = [dhol LpE9.34 
Bid = 1201, 
Bname = Bname OUTPUT, 
Bage = Bage OUTPUT 


SELECT Bneme as N'Bname', 
Bage as N'Bage' 


SELECT 'Return Value' = Breturn value 


le7\MssQLSERVER2008 (| WIDW-9527\Administrato, .| chapter12 | 00:00:00 | 1 行 
就 绪 第 ! 行 第 1 列 


图 12.22 运行 存储 过 程 pro_31 


通过 图 12.22 所 示 的 运行 效果 ， 读 者 可 以 对 比 本 章 的 示例 3， 看 看 有 什么 不 同 ? 对 比 
之 后 ， 读 者 会 发 现实 际 上 运行 的 效果 是 一 样 的 。 因 此 ， 如 果 读 者 忘记 了 运行 存储 过 程 的 命 
令 ， 可 以 直接 使 用 企业 管理 器 中 的 选项 来 运行 ! 


名 说明: 如 果 读者 觉得 在 图 12.18 所 示 的 界面 中 ， 填 入 存储 过 程 的 参数 有 些 麻烦 ， 也 怕 填 
错位 置 。 那么 , 可 以 在 菜单 栏 选择 “查询 ”|“ 指 定 模板 参数 的 值 ”选项 , 打开 “ 指 
定 模板 参数 的 值 ” 对 话 框 。 如 图 12.23 所 示 。 


指定 模板 参数 的 值 x| 


Author 

Create Date 

Description 
Procedure_Name 

Paran1l 
Datatype_For_Paranl 
Default_Yalue_For_Paranl 


Datatype_For_Paran2 
Default_Yalue For_Paran2 


图 12.23 ”指定 模板 参数 的 值 
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全 提示 : 在 图 12.23 所 示 界 面 中 ， 将 参数 值 填 入 相应 的 位 置 即 可 。 单 击 “ 确 定 ” 按 钮 ， 即 
可 保存 填 入 的 参数 信息 。 


12.5.2 ”使 用 企业 管理 器 修改 存储 过 程 


在 企业 管理 器 中 修改 存储 过 程 ， 相 对 于 创建 存储 过 程 就 容易 得 多 了 。 有 了 前 面 创建 存 
储 过 程 的 基础 ， 下 面 就 通过 示例 11 来 一 同 实践 如 何 修改 存储 过 程 。 

【示例 11】 修改 存储 过 程 pro_31， 添 加 一 个 输出 参数 一 一 性 别 。 

在 企业 管理 器 中 修改 存储 过 程 pro_ 31， 需 要 通过 以 下 5 个 步骤 完成 。 

(1) 在 “对 象 资源 管理 器 ”窗口 中 ， 展 开 chapter12 节点 。 

(2) 在 chapter12 的 节点 下 ， 找 到 “可 编程 性 ”节点 ， 并 将 其 展开 。 

(3) 在 “可 编程 性 ” |“ 存储 过 程 ” 节 点 下 ， 右 击 pro_31 节点 ， 在 弹出 的 右键 菜单 中 
选择 “修改 ”选项 ， 打 开 修 改 存储 过 程 界面 ， 如 图 12.24 所 示 。 


Rx Microsoft SQL Server Managemen tel El 
文件 () 编 得 ”视图 查 襄 Co) 项 目 P) ”调式 (0) 工具 (D 窗口 (Ww) 社区 (O 


帮助 (H) 


;也 新 建 查询 由 | 让 也 也 加 | 让 | 区 加 本 | 台大 

Wy | chapterlz |? 执行 0 nv 史 司 | 加 | ”起 | 好 因 | 训 

SQLQueryl.sql -istrator (53))* | 村 
GO 
/rennet C Object: Scoredprocedure [dbo] .[pro_31] Script Date: 
SET ANSI_NULLS ON 
Go 


SET QUOTED_IDENTIFIER ON 


日 ALTER PROCEDURE 


-- Add the p the stored procedure here 
Bid inc,Bname varchar (20] output,Bage int output 
aS 
BEGIN 
日 -- SET NOCOUNT ON added to prevent extra result sets from 


nterfering with SELECT statements. 
SET NOCOUNT ON; 


-- Insert statements for procedure here 
SELECT Bname=name,Bage=age FRON reginfo WHERE id=Bid; 


4 » 


于 已 .| wlDW-9527\M55QLSERVER2008(...| WIDW-9527\Administrato..，| chapter12 | 00:00:00 | 0 行 
就 绪 行 27 列 1 chl ms 


图 12.24 ”修改 存储 过 程 界面 


(4) 在 图 12.24 所 示 的 界面 中 ， 填 入 新 的 输出 参数 “@sex” 并 修改 查询 语句 。 修 改 后 
的 效果 如 图 12.25 所 示 。 

(5) 检查 图 12.25 修改 后 的 存储 过 程 ， 就 可 以 通过 选择 菜单 “查询 ”|“ 执 行 ” 选 项 ， 
来 执行 存储 过 程 修改 命令 ， 完 成 存储 过 程 的 修改 操作 。 

至 此 ， 存 储 过 程 就 修改 完成 了 ， 下 面 就 来 测试 修改 后 的 存储 过 程 执行 结果 是 否 正 确 。 
在 “可 编程 性 ”|“ 存 储 过 程 ” 节 点 下 ， 右 击 pro_31 节点 ， 在 弹出 的 右键 菜单 中 选择 “ 执 
行 存储 过 程 ” 选 项 ， 弹 出 图 12.26 所 示 界 面 。 
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Rroso SQ Server Monagenent Sudo 
文件 (E) ”编辑 (E) ”视力 (YW) 查询 (0) 项目 (P) 调式 (D0) 工具 (TD 窗口 (W) ”社区 (QO 才 助 (HH) 


:及 新 六 吉 询 YD | 让 | 配 诈 台 | 记 | 区 吕 马 | 虽 吕 
; 8 | chapterl2 "| 执行 WW 》 及 路 吝 
“SQLQuery4.sql -istrator (56))* 


50 


Author: <Author, , Name> 
Creare date: <Create Date,,> 
Description: <Description,,> 


日 ALTER PROCEDURE [dbo] . [pro_31] 
-- Add the parameters for the stored procedure here 
Bid int,Bname varchar (20) output,Bage int output, 
Bsex varchar (5) output 
a5 
BEGIN 
-- SET NOCOUNT ON added to prevent extra result sets from 
r -- interfering with SELECT statements. 
SET NOCOUNT ON; 


-- Insert statements for procedure here 
SELECT Bneme=neme,Bage=age, Bsex=sex FROM reginfo WHERE id=Bid; 


执行 过 程 


[dbol[pro_31] 


连接 有 
服务 器 
IDW-9527 \MSSQLSERVER2008 
连接 
TDW-9527\Adninistrator 


吉 ， 查 看 过 接 尾 性 


就 绪 


[El 


图 12.26 运行 存储 过 程 pro_31 界面 
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从 图 12.26 所 示 的 界面 中 ， 可 以 看 到 修改 后 的 pro_31 多 了 一 个 输出 参数 @sex。 这 里 ， 
仍然 给 @id 这 个 输入 参数 添 入 参数 值 “1201”， 并 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 运行 存储 过 
程 。 运 行 效果 如 图 12.27 所 示 。 


有 Microsoft SQL Server Management Studig -lolx| 
文件 E) ”编辑 (EB) ”视图 W 查询 (9) 项 目 (P) 调试 D) 工具 ID 

窗口 (W) 社区 (QO 帮助 

也 新 建 查询 六 | 镑 议 镶 | 让 | 苞 回 纪 | 加 s 

MY 二 | chapter12 | 执行 O 更 v 83 司 站 


SQL QueryS.sql -nistrator (57)) 
bsE [chapter1z] 
Go 


日 DECLARE Brecurn Value int, 
Bname varchar (20) 
Bage int, 
Basex varchar (5) 


HEXEC ~ Breturn value = [dbo] .[pro_31] 
Bid = 1201, 
Bname = Bname OUTPUT, 
Bage = Bage OUTPUT, 
Baex = Baex OUTPUT 


向 SELECT Bname as N'Bname', 
Bage as N'Bage', 


Bsex as N'Bsex' 


SELECT 'Return Value' = Breturn Value 


VER2008 (,.，| WIDW-9527WAdministrato,,，| chapter12 | 00:00:00 | 2 行 
行 1 列 1 Chl 


图 12.27 运行 pro_31 存储 过 程 的 效果 


12.5.3 ”使 用 企业 管理 器 删除 存储 过 程 


删除 存储 过 程 可 谓 又 是 所 有 存储 过 程 操 作 最 简单 的 一 项 了 ， 因 此 ， 现 在 可 以 舒缓 一 下 
心情 ， 轻 松 学 习 了 。 下 面 使 用 示例 12 来 学 习 如 何在 企业 管 器 中 删除 存储 过 程 。 

【示例 12】 在 企业 管理 器 中 ， 删 除 存储 过 程 pro_31。 

在 企业 管理 器 中 ， 删 除 存储 过 程 需要 通过 以 下 两 个 步骤 完成 。 

(1) 在 “对 象 资源 管理 器 ”窗口 中 ， 依 次 展开 “数据 库 ”|chapter12|“ 可 编程 性 ”|“ 存 
储 过 程 ”节点 ， 右 击 需要 删除 的 存储 过 程 ， 在 弹出 的 右键 菜单 中 选择 “删除 ”选项 ， 弹 出 
“删除 对 象 ”对 话 框 ， 如 图 12.28 所 示 。 

(2) 在 图 12.28 所 示 界 面 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 将 存储 过 程 pro_31 删除 了 。 


全 注意 : 在 删除 存储 过 程 后 ， 如 果 其 他 的 对 象 引 用 该 存储 过 程 ， 那 么 在 运行 时 就 会 出 现 错 
误 。 因 此 ， 在 删除 存储 过 程 前 ， 最 好 在 图 12.28 所 示 界 面 中 单 击 “ 显 示 依赖 关系 ” 
按钮 来 查看 是 否 有 其 他 对 象 使 用 该 存储 过 程 。 


“22359. 


服务 器 
WIDN-9527 \NSSQLSERYER2008 


连接 
WIDN-9527\Adninistrator 


圣 二 看 连接 屋 性 
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在 本 章 中 主要 讲解 了 存储 过 程 的 作用 、 创 建 、 修 改 以 及 删除 。 在 创建 存储 过 程 部 分 ， 
主要 讲解 了 不 带 参数 和 带 参数 存储 过 程 的 创建 与 执行 方法 ; 在 修改 存储 过 程 部 分 ， 主 要 讲 
解 了 使 用 ALTER 语句 修改 存储 过 程 以 及 给 存储 过 程 重 命名 ; 在 删除 存储 过 程 部 分 ， 分 别 
讲解 了 删除 1 个 和 多 个 存储 过 程 。 此 外 ， 在 本 章 中 还 讲解 了 如 何 使 用 企业 管理 器 的 图 形 界 


面 来 创建 和 管理 存储 过 程 。 


图 12.28 删除 存储 过 程 对 话 杠 


12.6 本 章 小 结 


12.7 本 章 习 题 


一 、 填 空 题 

1. 系统 存储 过 程 的 名 称 通常 是 以 为 前 级 的 。 

2. 存储 过 程 中 的 参数 类 型 有 种 ， 分 别 是 。 
3. 创建 带 加 密 选项 的 存储 过 程 需要 使 用 的 语句 是 
二 、 选 择 题 

1. 修改 存储 过 程 名 称 的 语句 是 


A. CREATE 语句 


。260 。 


B. ALTER 语句 
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C. SP_RENAME 语句 D. 以 上 都 不 是 
2. 存储 过 程 的 类 型 包括 ” ___。 

A. 系统 存储 过 程 B. 自 定义 存储 过 程 

C. 扩展 存储 过 程 D. 以 上 都 是 
3. 执行 存储 过 程 语句 是 。 

A. USE 语句 B. EXEC 语句 C. DO 语句 D. 以 上 都 不 是 
三 、 问 答题 


1. 存储 过 程 有 哪些 优势 ? 
2. 使 用 什么 语句 查看 存储 过 程 创建 的 语句 ? 
3. 在 什么 情况 下 使 用 存储 过 程 中 的 输出 参数 ? 


四 、 操 作 题 


试 着 创建 一 个 INSTEAD OF 存储 过 程 ， 当 表 执行 添加 操作 时 ， 将 删除 该 表 中 的 全 这 
数据 。 


ms 
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提 到 确保 数据 完整 性 ， 读 者 是 否 想起 在 前 面 的 章节 中 学 习 过 的 约束 呢 ? 没 错 ， 约 束 是 
一 种 保证 数据 完整 性 的 方法 。 实 际 上 ， 在 数据 库 中 还 有 另 一 种 方法 来 确保 数据 的 完整 性 ， 
那 就 是 本 章 要 学 习 的 触发 器 了 。 

章 的 主要 知识 点 如 下 : 

口 触发 器 的 作用 和 分 类 

口 如 何 创 建 触发 器 

口 如 何 修改 触发 器 

口 如 何 删除 触发 器 


13.1 意思 的 触发 器 


触发 器 与 存储 过 程 不 同 ， 它 不 需要 使 用 EXEC 语句 调用 就 可 以 执行 。 但 是 ， 在 触发 器 
中 所 写 的 语句 又 与 存储 过 程 类 似 ， 因 此 ， 经 常会 把 触发 器 看 作 是 一 种 特殊 的 存储 过 程 。 触 
发 器 可 以 在 对 表 进 行 UPDATE、INSERT 和 DELETE 这 些 操作 时 ， 自 动 地 被 调用 。 


13.1.1 触发 器 的 作用 


所 谓 知己 知 彼 ， 百 战 不 列 ， 知 道 了 触发 器 的 作用 才能 够 在 应 用 的 时 候 有 的 放 矢 。 触 发 
器 最 重要 的 作用 就 是 能 够 确保 数据 的 完整 性 ， 但 同时 也 要 注意 每 一 个 数据 操作 只 能 设置 一 
个 触发 器 。 

触发 器 的 执行 可 以 通过 数据 表 中 数据 的 变化 来 触发 ， 也 就 是 说 当 向 表 插入 一 条 数据 
时 ， 和 触发 器 就 会 知道 。 那 么 ， 如 果 要 禁止 向 表 中 添加 数据 ， 就 可 以 及 时 地 来 控制 对 表 的 操 
作 。 另 外 ， 当 检测 到 某 张 数据 表 中 数据 变化 时 ， 还 可 以 及 时 更 新 其 他 数据 表 ， 并 能 够 获取 
到 更 新 的 数据 。 但 是 ， 在 数据 库 中 如 果 触 发 器 过 多 也 会 影响 数据 库 的 效率 ， 因 此 ， 在 数据 
库 中 要 合理 地 使 用 触发 器 ， 及 时 删除 不 用 的 触发 器 。 


13.1.2 触发 器 分 类 


在 SQL Server 数据 库 ， 触 发 器 主要 分 为 3 大 类 ， 即 登录 触发 器 、DML 触发 器 和 DDL 
触发 器 。 本 章 中 主要 讲解 DML 类 型 的 触发 器 。 但 是 ， 其 他 类 型 的 触发 器 也 是 需要 了 解 的 。 
下 面 就 将 这 3 类 触发 器 的 主要 作用 说 明 如 下 。 

口 登录 触发 器 : 它 是 作用 在 LOGIN 事件 的 触发 器 ， 是 一 种 AFTER 类 型 触发 器 ( 表 

示 在 登录 后 激发 )。 使 用 登录 触发 器 可 以 控制 用 户 会 话 的 创建 过 程 以 及 限制 用 户 名 
和 会 话 的 次 数 。 


口 
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DML 和 触发 器 : 它 包 括 对 表 或 视图 DML 操作 激发 的 触发 器 。DML 操作 包括 
UPDATE、INSERT 或 DELETE 语句 。DML 触发 器 包括 两 种 类 型 的 触发 器 ， 一 种 
是 AFTER 类 型 ， 一 种 是 INSTEAD OF 类 型 。AFTER 类 型 表示 对 表 或 视图 操作 完 
成 后 激发 触发 器 ; INSTEAD OF 类 型 表示 当 表 或 视图 执行 DML 操作 时 ， 奉 代 这 些 
操作 执行 其 他 一 些 操作 。 

DDL 触发 器 : 它 包 括 对 数据 库 对 象 执行 DDL 操作 后 激发 的 触发 器 。DDL 操作 包 
括 CREATE、ALTER 和 DROP 等 。 该 触发 器 一 般 用 于 管理 和 记录 数据 库 对 象 的 结 
构 变 化 。 


13.2 ”创建 触发 器 


创建 触发 器 是 开始 使 用 触发 器 的 第 一 步 , 有 了 这 重要 的 一 步 , 才 可 以 完成 后 续 的 操作 。 
创建 触发 器 可 以 使 用 SQL 语句 也 可 以 通过 企业 管理 器 中 的 图 形 界面 来 操作 , 在 本 节 中 主要 
讲解 使 用 SQL 语句 创建 触发 器 。 


13.2.1 


创建 触发 器 的 语法 


在 创建 触发 器 时 ， 使 用 的 是 CREATE TRIGGER 语句 。 这 里 ， 给 出 的 是 创建 DML 触 
发 器 的 语法 ， 也 是 本 章 要 研究 的 重点 。 有 具体 的 语法 形式 如 下 : 


CREATE TRIGGER trigger _ name 

ON { table | view } 

[ WITH ENCRYPTION] 

{ FOR | AFTER | INSTEAD OF } 

SINSPRE Nl DATE | DET 
[ NOT FOR REPLICATION ] 

RS { sql statement } 


其 中 : 


口 
口 
口 
口 
口 


口 


口 
口 


trigger_name: 触发 器 的 名 称 。 

table | view: 触发 器 作用 的 表 名 或 视图 名 。 

WITH ENCRYPTION: 对 文本 进行 加 密 。 与 它 在 存储 过 程 中 的 含义 一 样 。 
FORIAFTER: 当 执 行 某 些 操作 后 被 激发 。 比 如 : 向 表 中 添加 数据 后 激发 。FOR 与 
AFTER 是 同 义 的 。 

INSTEAD OF: 替代 操作 ， 需 要 注意 的 是 对 于 表 或 视图 ， 每 个 INSERT、UPDATE 
或 DELETE 语句 最 多 可 定义 一 个 INSTEAD OF 触发 器 。 

{ [DELETE] [,] [INSERT] [,] [UPDATE] }: 指定 在 哪 种 操作 时 激发 触发 器 。 可 以 选 
择 1 到 多 个 选项 。 

NOT FOR REPLICATION: 当 复 制 表 时 ， 触 发 器 不 被 激发 。 

sql_statement: 触发 器 被 激发 时 执行 的 工 SQL 语句 。 


全 说 明 : 在 DML 触发 器 中 ， 不 能 够 在 sql_statement 部 分 写 DDL 语句 。 
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13.2.2” 建 AFTER 类 型 触发 器 


有 了 上 面 的 语句 ， 就 可 以 尽情 地 演绎 触发 器 了 。 首 先 要 学 习 的 是 第 一 种 类 型 的 触发 
器 AFTER 触发 器 ， 它 通常 是 在 表 进 行 了 某 项 操作 之 后 ,激发 触发 器 。 在 做 触发 器 的 相 
关 练 习 之 前 , 还 要 准备 本 章 中 要 使 用 的 数据 库 和 数据 表 。 本 章 中 使 用 的 数据 库 是 chapter13， 
在 该 数据 库 需 要 创建 3 张 数据 表 ， 分 别 是 图 书信 息 表 (bookinfo)、 借 阅 信息 表 (readinfo) 
和 读者 信息 表 (userinfo)。 表 结构 分 别 如 表 13-1、 表 13-2 和 表 13-3 所 示 。 


表 13-1 图 书信 息 表 (bookinfo) 


序号 列 名 数据 类 型 说 了 明 
1 bookid int 图 书 编号 
2 bookname varchar(50) 图 书 名 称 
3 bookprice decimal(6,2) 图 书 价格 
4 bookpub varchar(30) 出 版 社 
5 bookauthor varchar(20) 作者 
6 bookcount int 馆藏 数量 


表 13-2 借阅 信息 表 (readinfo) 


借阅 流水 号 
借阅 人 编号 
借阅 图 书 编号 


序号 列 名 数据 类 型 说 明 
1 userid int 读者 编号 
2 Username varchar(20) 读者 姓名 


使 用 CREATE TABLE 语句 或 者 是 直接 在 企业 管理 器 中 创建 好 这 3 张 表 ， 然 后 按照 
表 13-4 和 表 13-5 的 内 容 分 别 向 图 书信 息 表 和 读者 信息 表 中 添加 数据 。 
表 13-4 图书 信息 表 中 的 数据 
图 书 名 称 
计算 机 基础 
SQL 基础 
C# 编 程 


线性 代数 


好 了 ， 数 据 准 备 工 作 就 到 此 结束 了 。 现 在 就 要 开始 演练 了 。 
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【示例 1】 创建 AFTER 触发 器 ， 当 “ 王 明 ” 这 个 读者 借阅 “SQL 基础 ”这 本 书 时 ， 
将 图 书信 息 表 (bookinfp) 中 “SQL 基础 ”这 本 书 的 馆藏 数量 减 1。 

根据 题目 要 求 ， 是 在 向 借阅 信息 表 中 添加 一 条 数据 后 ， 也 就 是 在 INSERT 命令 后 激发 
触发 器 ， 修 改 图 书信 息 表 的 信息 。 具 体 语句 如 下 : 


CREATE TRIGGER TRI 1 

ON readinfo 

AFTER INSERT 

AS 

UPDATE bookinfo SET bookcount=bookcount-1 where bookid=1002; 


执行 上 面 的 语句 ， 效 果 如 图 13.1 所 示 。 


Microsoft SQL Server Management Studio =|Dlxl 
文件 编辑 (B 视图 查询 (Q) 项 目 (BE) 调试 D) 工具 (D 窗口 (Ww) 社区 (Q 帮助 (HD) 
卫 新 建 查 尖山 也 也 四 | 记 | 盛 四 己 | 总 天 
曙 天 | deptors "|? 执 生 W PR v 弹 许 回 | 六 动 | 她 圈 引 
SQLQuery1.sql -istrator (53))* | vx 
BCREATE TRIGGER TRI_1 

ON readinfo 

AFTER INSERT 

35 

UPDATE bookinfo SET bookcount=bookcount-1 where bookid=1002; 


| WIDW-9527\MS5QLSERYER2008 (| WIDW-9527\Administrato.,，| chapter13 | 00:00:00 | 0 行 
行 9 列 1 Chl 


OE 


图 13.1 创建 触发 器 tri_1 
下 面 就 按照 题目 要 求 看 看 触发 器 是 否 起 作用 了 。 添 加 语句 如 下 : 


INSERT INTO readinfo VALUES (2, 1002); 


执行 上 面 的 语句 ， 效 果 如 图 13.2 所 示 。 

从 图 13.2 所 示 的 效果 ， 可 以 看 出 有 2 条 “(1 行 受 影响 )” 的 消息 。 这 就 说 明 ， 不 仅 执 
行 了 在 查询 编辑 器 中 编写 的 语句 ， 还 执行 了 其 他 的 语句 。 那 么 ， 执 行 的 其 他 语句 就 是 触发 
器 中 所 写 的 语句 。 在 触发 器 中 执行 的 SQL 语句 是 对 图 书信 息 表 〈bookinfo) 所 做 的 修改 ， 
下 面 就 来 查看 图 书信 息 表 ， 看 看 更 改 后 的 效果 。 如 图 13.3 所 示 。 


| 
文件 E) ”篇 辑 (视图 W 查询 (Q) 项 目 E) 调试 D) 工具 (D 窗口 (w) 
社区 (QO 帮 盈 (HH) 
和 2 | 壕 渤 钙 | 忆 | 扇 回 寺 加 ; 
本 38 | hapter13 ~| ?执行 9 有 v ?3 时 [ 国 


卫 闽 建 查询 W | 出 | 轧 也 配 | 负 | 蕊 加 要 怠 志 
a 3 | chapter13 "| 执行 WW 》 和 路 疗 久 


SQ Queryl.sql .istrator (53))* 
日 INSERT INTO readinfo VALUES (2, 1002]; 


7 | 
人 1 行 受 影响 ; 
人 1 行 受 影响 ) 
出 
‘55QLSERYER2008 (… | WIDW-9527Wdmingtrato..，| chapter13 | 00:00:00 | 0 行 
就 结 行 3 列 1 Chi 


图 13.2 ”向 表 中 添加 数据 后 激发 触发 器 的 效果 图 13.3 图 书信 息 表 的 数据 


“9527\M55QL SERVER2008 (... | WIDW-9527Administrsto.,，| chapter13 | 00:00:00 | 5 行 
就 绪 行 4 列 1 hl pp 
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从 图 13.3 所 示 的 效果 ， 可 以 看 出 “SQL 基础 ”这 本 书 的 数量 (bookcount) 少 了 1。 也 


就 是 说 前 面 创建 的 触发 器 tri_1 被 触发 了 。 


从 示例 1 所 创建 的 触发 器 tri_1 可 以 看 出 , 触发 器 的 作用 太 单一 了 , 没有 有 具体 的 实际 作 


用 。 那 么 ， 如 何 能 做 到 只 要 借阅 信息 表 中 添加 数据 ， 就 更 改 相 应 的 馆藏 图 书 数量 。 下 面 就 


是 向 读者 隆重 推出 触发 器 中 常用 的 一 张 临时 表 inserted。inserted 这 张 临时 表 中 存放 的 就 是 
在 成 功 执行 INSERT 或 UPDATE 语句 后 ， 将 被 插入 或 更 新 的 值 。 因 此 ，INSERTED 这 张 表 
可 是 在 触发 器 中 使 用 的 法 宝 啊 。 下 面 就 将 示例 1 修改 成 根据 图 书 借阅 信息 表 中 数据 的 增长 
来 更 新 图 书信 息 表 中 图 书 的 馆藏 数量 。 


【示例 2】 创建 触发 器 tri 2， 当 图 书 借阅 信息 表 中 有 数据 增加 时 ， 就 更 改 相应 的 图 书 


信息 表 中 的 馆藏 数量 。 


根据 题目 要 求 ， 使 用 inserted 临时 表 获 取 新 添加 的 图 书 编号 ， 然 后 再 更 新 图 书信 息 表 。 


基体 语句 如 下 : 


CREATE TRIGGER tri 2 

ON readinfo 

AFTER INSERT 

RS 

BEGIN 

DECLARE Q@bookid int; -声明 变量 存储 图 书 编号 

SELECT Q@bookid=bookid FROM inserted;-- 从 inserted 表 中 查询 出 新 添加 的 图 书 编号 
UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=@bookid; 

END 


执行 上 面 的 语句 ， 效 果 如 图 13.4 所 示 。 


a Microsoft SQL Server Management Studio 1 -Iolxl 
文件 (E) ” 蝙 辑 (E) ”视图 (W 查询 (9) 项目) 调试 D) 工具 (D 窗口 (Ww) 社区 (CO 帮助 (H) 
及 新 寻 查 注册] 多 六 串 | 已 | 攻 回 对 | 妥 忆 
; MY 2 | chapter13 *|? 执 5 有 v 吕 本国 | 小强 | 的 央 约 | 三 全 | 素 名 
SQLQuery2.sql .istrator (52))* | 下 
日 CREATE TRIGGER tri 2 
ON readinfo 
AFTER INSERT 
AS 
BEGIN 
DECLARE Bbookid int; ”-- 声 明 变量 存储 图 书 编号 
SELECT Bbookid=bookid FRON inserted; -- 从 inserted 表 中 查询 出 新 添加 的 图 书 编号 
UPDATE bookinfo SET bookcounc=bookcounc-1 WHERE bookid=Bbookid; 
END 


局 消息 | 

命令 已 成 功 完成 。 = 
本 
园 查询 已 成 功 执行 。 。 | WIDW-9527WIM5SQLSERYER2008 (… | WIDW-9527VAdministrato..，| chapter13 | 00:00:00 | 0 行 
就 绪 行 11 列 1 Chl Ins 


图 13.4 创建 触发 器 tri 2 
下 面 就 来 验证 tri_2 触发 器 是 否 满足 了 题目 要 求 。 向 借阅 信息 表 (readinfo) 中 添加 一 


条 数据 ， 并 查询 图 书信 息 表 (bookinfo〉 中 的 数据 。 这 里 ， 读 者 要 注意 的 是 目前 数据 库 中 
有 两 个 触发 器 都 是 在 向 表 readinfo 中 添加 数据 后 激发 的 ， 因 此 ， 需 要 先 将 触发 器 tri_1 删除 
掉 ， 否 则 ， 就 不 会 执行 新 创建 的 触发 器 tri 2 了 。 语 名 如 下 : 


DROP TRIGGER tri 1; -删除 触发 器 tri_1 
INSERT INTO readinfo VALUES (2, 1003); 
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SELECT * FROM bookinfo; 


执行 上 面 的 语句 ， 效 果 如 图 13.5 所 示 。 


Eicrosoft SOL Server Monagement Studio SF 
文件 (编辑 (E) 视图 查询 (o) 项 目 (E) 调 坛 D) 工具 (D 窗口 (Ww) 
社区 (CO 帮助 

也 新 寻 查询 | 记 | 轧 遍 可 | 记 | 写 加 己 | 带 所 

a 3 | chapterl3 "| 执行 W) 》 则 V 吕 癌 
“SQL Query2.sql .istrator (52))* 


‘9527\M55QLSERVER2008 (.,，| WIDW-9527\Administrato.,，| chapter13 | 00:00:00 | 5 行 


就 绪 第 1 行 第 1 列 


图 13.5 激发 触发 器 tri_2 的 效果 


从 图 13.5 所 示 的 执行 效果 可 以 看 出 ，tri_2 确实 是 将 图 书 编号 是 1003 的 馆藏 数量 减 
Ve 

读者 可 以 需要 思考 这 样 一 个 问题 了 ， 如 果 图 书 的 馆藏 数量 为 0， 那 么 ， 图 书 还 可 以 借 
阅 吗 ? 如 何 通过 触发 器 解决 这 样 的 问题 呢 ? 没 错 ， 这 就 需要 在 借阅 图 书 前 判断 所 借 图 书 的 
馆藏 数量 是 否 为 0， 如果 为 0 就 不 能 够 借阅 。 同 时 ， 还 要 在 触发 器 中 用 到 事务 。 下 面 就 一 
同 来 学 习 示 例 3 吧 。 

【示例 3】 创建 触发 器 tri_ 3， 在 示例 2 的 基础 上 添加 判断 ， 如 果 借 书 时 图 书 的 馆藏 数 
量 为 0， 则 不 可 以 借阅 ， 否 则 正常 借阅 。 

根据 题目 要 求 ， 语 句 如 下 : 

CREATE TRIGGER tri 3 

ON readinfo 

AFTER INSERT 

AS 

BEGIN 

DECLARE @bookid int,@count int; -声明 变量 存储 图 书 编号 和 馆藏 数量 

SELECT Qbookid=bookid FROM inserted; -- 从 inserted 表 中 查询 出 新 添加 的 图 书 编号 

SELECT Qcount=bookcount FROM bookinfo WHERE bookid=@bookid; 

-- 从 图 书信 息 表 中 查询 出 馆藏 图 书 数量 

if(@count=0) -判断 馆藏 图 书 数量 是 否 为 0 

BEGIN 

ROLLBACK TRANSACTION; =-- 回 滚 事 务 

END 

ELSE 

BEGIN 

UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=@bookid; 


END 
END 


执行 上 面 的 语句 ， 效 果 如 图 13.6 所 示 。 
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十 
文件 日” 蝙 辑 (E) 视图 ( ”查询 (9) 项 目 (P) 调试 D) 工具 (D 窗口 (Ww) 社区 (QO ”帮助 (H) 
对 | 六 | 辣 渤 钙 | 店 | 蕊 回 对 | 加 局 
;一 女 | bopters -| ?Ww ha v 时 名 国 | 六 马 | 色 图 如 | 三 全 | 地 这 | 名 电 
WIIDW-9527WM55.， dbo,bookinfo “5QLQuery2.sql .istrator (52))* | -x 
日 CREMTE TRIGGER tri 3 
ON readinfo 
AFTER INSERT 
25 
BEGIN 
| DECLARE Bbookid int,Bcount int; -- 声 明 变 量 存储 图 书 编号 ,馆藏 数量 
SELECT Qhookid=bookiq FRON inserted; -- 从 inserted 表 中 查询 出 新 添加 的 图 书 编号 
SELECT Bcount=bookcount FRONM bookinfo WHERE bookid=Bbookid; -- 从 图 书信 息 表 中 查询 出 馆藏 图 书 数量 
证 (Bcounr-0 ~- 判断 馆藏 图 书 数量 是 否 为 0 
BEGIN 
ROLLBACK TRANSACTION; -- 回 流 事 务 
END 
ELSE 
BEGIN 
UPDATE bookinfo SET bookcounc-bookcounc-1 WHERE bookid-Bhookid; 
END 
-END 


| WIDW-9527\M55QLSERVER2008 (.，| WIDW-9527\Administrato.,。| chapterl3 | 00:00:00 | 0 行 
行 !9 列 ! ahi 


图 13.6 创建 触发 器 tri_3 


下 面 就 将 图 书信 息 表 中 图 书 编号 为 1002 的 图 书馆 藏 数量 改写 成 0, 并 执行 一 条 借阅 图 
书 1002 的 语句 ， 来 验证 触发 器 tri_3 是 否 正确 。 这 里 仍然 需要 注意 的 是 需要 将 之 前 创建 的 
tri_ 2 触发 器 删除 。 语 句 如 下 : 


DROP TRIGGER tri 2; 
UPDATE bookinfo SET bookcount=0 WHERE bookid=1002; 
INSERT INTO readinfo VALUES (2, 1002); 


执行 上 面 的 语句 ， 效 果 如 图 13.7 所 示 。 
Rvs Sa seme orooencnt Sted = 


文件 (E) ”编辑 (E) ”视图 (V) 查询 (Q) 项目) 调试 D) 工具 D 窗 D(w) 社区 (QO 


帮助 tD) 
及 | 访 | 志 证 怨 | 让 | 蕊 回 当 | 加 
By B82 | chapter13 -| 2 执 SW) 由 w ?3 本 


dbo.bookinfo SQL Query2.sql istrator (52))* 
下 DROP TRIGGER tri 2; 

UPDATE bookinzo SET bookcount=0 WHERE bookid=1002; 
INSERT INTO readinfo VALUES (2, 1002); 


人 1 行 受 影响 ) 
消息 3609， 级 别 16， 状 态 1, 第 2 行 
事务 在 甬 发 器 中 结束 。 批 处 理 已 中 止 。 


WIDW-9527WM55QLSERVER2008 (..,| WIDW-9527\Administrato,.，| chapter13 | 00:00:00 | 0 行 | 
匹配 : BEGIN chi 


图 13.7 验证 触发 器 tri_ 3 


从 图 13.7 所 示 的 结果 ， 可 以 看 出 触发 器 中 的 事务 拒绝 了 向 表 readinfo 中 插入 信息 的 
请 求 。 

通过 前 面 的 3 个 示例 ， 基 本 上 可 以 完成 图 书 的 借阅 操作 了 ， 但 是 ， 当 读者 要 还 书 的 时 
候 ， 又 应 该 如 何 处 理 呢 ? 在 还 书 的 时 候 ， 这 里 要 求 将 读者 借阅 的 相应 的 图 书信 息 删 除 。 那 
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么 ， 删 除 借阅 信息 后 ， 如 何 使 用 触发 器 将 归还 的 图 书 数量 加 到 图 书信 息 表 中 的 馆藏 数量 中 
呢 ? 前面 给 读者 介绍 过 临时 表 inserted, 它 是 用 来 存放 添加 或 更 新 的 数据 的 。 那么 , 这 回 是 
删除 数据 操作 ， 删 除数 据 操作 时 将 删除 掉 的 数据 存放 到 另 一 张 临时 表 DELETED 中 。 这 回 
读者 有 了 前 面 示 例 的 基础 ， 应 付 还 书 时 更 新 馆藏 数量 的 触发 器 还 是 能 够 得 心 应 手 吧 。 下 面 
就 一 同 来 完成 示例 4 吧 。 

【示例 4】 创建 触发 器 tri_ 4， 实现 当 还 书 时 更 新 图 书信 息 表 中 的 馆藏 数量 。 

根据 题目 要 求 ， 要 使 用 临时 表 DELETED 完成 。 具 体 语句 如 下 : 


CREATE TRIGGER tri 4 

ON readinfo 

AFTER DELETE 

RS 

BEGIN 

DECLARE Q@bookid int; 

SELECT @bookid=bookid FROM deleted; 

UPDATE bookinfo SET bookcount=bookcount+1 WHERE bookid=@bookid; 
END 


执行 上 面 的 语句 ， 效 果 如 图 13.8 所 示 。 


本 Microsoft 5QL Server Management Studio =|DIxl 
文件 目 ”编辑 (E) ”视图 (Y) 查询 (Q) 项目) 汤 式 (0) 工具 (D 窗口 (W) 社区 (QO 帮助 (H) 

也 新建 查 询 N) 六 | 六 六 咏 | 店 | 区 加 对 | 癌 

Ry 2 | chapter!3 -| 1 执 和 0 有 v 吕 名 国 | 下 弄 | 的 图 扩 

” WIDW-9527\MS5,., dbo,bookinfo SQLQuery2.sql .istrator (52))* vx 
日 CREATE TRIGGER tri 4 


ON readinfo 
AFTER DELETE 


DECLARE Bbookid int; 

SELECT Bbookid=bookid FROM deleted; 

UPDATE bookinfo SET bookcount=bookcount+1 WHERE bookid=Bbookid; 
END 


加 消息 | 
命令 已 成 功 完成 。 本 
日] 


园 查询 已 成 | wIDW-9527WMS5QLSERYER2006 (.,，| WIDW-9527\Administrato,,，| chapter13 | 00:00:00 | 0 行 
匹配 ; BEGIN 行 11 列 1 ch1l Ins 


图 13.8 ”创建 触发 器 tri 4 
下 面 就 来 验证 新 创建 的 成 果 tri 4 吧 。 语 句 如 下 


SELECT * FROM bookinfo; -- 查 询 未 更 新 前 的 效果 
DELETE FROM readinfo WHERE bookid=1002; 
SELECT * FROM bookinfo; 一 -查询 更 新 后 的 效果 


执行 上 面 的 语句 ， 效 果 如 图 13.9 所 示 。 

通过 前 面 4 个 示例 ， 基 本 上 使 用 触发 器 完成 了 借 书 和 还 书 时 对 图 书信 息 表 中 图 书馆 藏 
数量 的 更 新 操作 。 下 面 就 用 示例 $ 将 前 面 4 个 示例 进行 整合 。 

【示例 5】 创建 触发 器 tri 5， 完 成 借 书 和 还 书 时 对 图 书信 息 表 中 图 书馆 藏 数量 的 更 新 
操作 。 

根据 题目 要 求 ， 借 书 是 对 借阅 信息 表 的 添加 操作 ， 还 书 是 对 借阅 信息 表 的 删除 操作 。 
因此 ， 触 发 器 在 创建 时 需要 基于 INSERT 和 DELETE 两 个 操作 。 具 体 的 语句 如 下 : 
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[mooson Sat Server Management Sodio 二 
文件 E) 编辑 (E) 视图 ( 男 ”查询 (9O) 项 目 B) 调试 D) 工具 (D 窗 D ”社区 (QO 帮助 (H) 

也 新 寻 查询 出 | 目 | 轧 也 加 | 记 蕊 回忆 到 下 

?2 | chapterl3 -| ?执行 9 有 v 品名 国 |” 续 | 镁 关 扩 
WIDW-9527\M55,.. dbo， “5QL Query2.sql -istrator (52))* 
日 SELECT * 了 RON Kinfo; 


| DELETE FRON readinfo WHERE bookid=1002; 
| SELECT * FROM bookinfo; 


回 结果 | 已 消息 | 


bookid 


,| WIDW-9527WAdministrato,,，| chapter13 | 00:00:00 | 10 行 
行 5 列 1 Chi 


匹配 : BEGIN 


图 13.9 触发 器 执行 效果 


CREATE TRIGGER tri 5 
ON readinfo 
AFTER INSERT,DELETE 


RS 

BEGIN 

DECLARE @bookid int,@count int; -- 声 明 变量 存储 图 书 编号 和 馆藏 数量 
if EXISTS (SELECT bookid FROM inserted) 

BEGIN 


SELECT Qbookid=bookid FROM inserted -- 从 :inserted 表 中 查询 出 新 添加 的 图 书 编号 
SELECT @count=bookcount FROM bookinfo WHERE bookid=@bookid; 


-- 从 图 书信 息 表 中 查询 出 馆藏 图 书 数量 

if(ecount=0) -- 判 断 馆藏 图 书 数量 是 否 为 0 

BEGIN 

ROLLBACK TRANSACTION; =-- 回 滚 事 务 

END 

ELSE 

BEGIN 

UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=@bookid; 
=-- 馆 藏 图 书 数量 减 

END 

END 

了 ELSE 

BEGIN 


SELECT @bookid=bookid FROM deleted; 
UPDATE bookinfo SET bookcount=bookcount+1 WHERE bookid=@bookid; 


=-- 馆 藏 图 书 数量 加 
END 
END 


执行 上 面 的 语句 ， 效 果 如 图 13.10 所 示 。 
至 此 ， 一 个 完整 的 图 书 借 还 藏书 数量 变化 的 触发 器 就 完成 了 。 下 面 就 到 了 检验 触发 器 
的 时 候 了 。 在 检验 触发 器 之 前 ,需要 将 之 前 创建 的 tri 4 触发 器 删除 。 因 为 每 一 个 数据 库 操 
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a Microsoft SQL Server Management Studio | x| 


文件 昌 。 编 加 四 视 加 查询 00) 项 目 D 调式 D) 工具 D 窗 D 社区 (OO 大 且 中 
卫 旨 Ra 林 | 记 | 蕊 也 可 | 启 | 克 回忆 | 骂 届 
a WR | chepterl3 "|! 5wW 》 mv 品名 国 | 站 鸡 | 的 立 拉 | 三 全 | 李 这 | 名 局 


-END 了 
1 » 


SQLQuery1.sql -istrator (52))*| “Xx 
章 CREATE TRIGGER tri 5 

ON readinfo 

AFTER INSERT, DELETE 

5 

BEGIN 

DECLARE Bhookid int,Bcount int; 一 声明 变量 存储 图 书 编号 和 馆藏 数量 

if EXISTS(SELECT bookid FROM inserted) 

BEGIN 

SELECT Bhookid=bookid FROM inserted -- 从 inserted 表 中 查询 出 新 添加 的 图 书 编号 

SELECT Bcount=bookcount FROM bookinfo WHERE bookid=Bbookid; ”-- 从 图 书信 息 表 中 查询 出 馆藏 图 书 数量 

if (Bcount=0) -- 判 断 馆藏 图 书 数量 是 否 为 0 

BEGIN 

ROLLBACK TRANSACTION; -- 回 液 事 务 

END 

ELSE 

BEGIN 

UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=Bbookid;  -- 人 馆藏 图 书 数量 减 1 

END 


SELECT Bbookid=bookid FRON deleted; 
UPDATE bookinfo SET hookcount=bookcount+1 WHERE bookid=Bbookid; -- 馆 藏 图 书 数量 加 1 
END 


加 消息 | 

命令 已 成 功 完成 。 = 
加 ， 
加 查询 已 成 功 执行 。 | WiDW-9527WMSSQLSERVER2008 (... | WIDW-9527\Administrato.,，| chapter13 | 00:00:00 | 0 行 


就 绪 行 26 列 ! Chl Ins .4 


图 13.10 创建 触发 器 tri_5 


作 只 能 有 一 个 触发 器 。 具 体 的 验证 语句 如 下 : 


DROP TRIGGER tri 4; -删除 触发 器 tri_4 

SELECT * FROM bookinfoy -- 查 询 图 书信 息 表 的 原始 数据 

INSERT INTO readinfo VALUES (1,2001); 

SELECT * FROM bookinfo; -- 查 询 借阅 图 书后 ,图书 信息 表 中 的 数据 
DELETE FROM readinfo WHERE userid=1 AND bookid=2001; 

SELECT * FROM bookinfo; -- 查 询 还 书后 ， 图 书信 息 表 中 的 数据 


执行 上 面 的 语句 ， 效 果 如 图 13.11 所 示 。 
通过 图 13.11 所 示 的 结果 ， 可 以 看 出 tri_5 触发 器 确实 能 够 完成 借 还 图 书馆 藏 数量 的 变 


化 的 操作 。 
13.2.3 再建 INSTEAD OF 类 型 触发 器 


难 习 


有 了 创建 AFTER 类 型 触发 器 的 基础 ， 创 建 INSTEAD OF 类 型 的 触发 器 就 不 是 什么 
和 了。 只 要 清楚 INSTEAD OF 触发 器 是 替代 原 有 操作 的 ， 比 如 : 删除 数据 表 时 激发 触 


发 器 使 其 向 表 中 添加 一 条 数据 。 下 面 就 通过 示例 来 学 习 如 何 使 用 INSTEAD OF 类 型 的 触 
发 器 。 


【示例 6】 创建 触发 器 tri 6， 当 读者 还 书 时 ， 不 将 图 书 借阅 信息 表 中 的 数据 删除 ， 但 


是 要 更 新 图 书信 息 表 中 的 馆藏 数量 。 
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也、 Microsoft SQL Server Management Studio =|Djxl 


文件 ”编辑 (E) 视图 查询 (9) ”项目 (E) 调式 D) 工具 (D 窗口 W ”社区 (QO 帮助 中 
PEE 


By Ha | chapter13 -| 1 执行 W 》 mv 弹 树 回 | 半 支 | 公 图 
SQLQuery1.sql .istrator (52))* | vx 
DROP TRIGGER tri 47 -- 删 除 触 发 回 tri_4 


书信 息 表 的 原始 数据 
INSERT INTO readinfo VALUES(1,2001); 


SELECT * FROK bookinfo; ”-- 查 询 借阅 图 书后 ， 图 书信 息 表 中 的 数据 
DELETE FROM readinfo WHERE userid=1 AND bookid=2001; 


SELECT * 了 RON bookinfo; ”~-- 查 询 还 书后 ， 图 书信 息 表 中 的 数据 


SELECT * FRONM bookinfo; -- 查 


园 喜 | wiDw-9527\M55QLSERVER2008(..，| WIDW-9527Vadministrato,,，| chapter13 | 00:00:00 | 15 行 
就 绪 行 8 列 1 chi Ins /4 


图 13.11 验证 触发 器 tri_5 
根据 题目 要 求 ， 需 要 创建 INSTEAD OF 类 型 的 触发 器 。 创 建 语句 如 下 : 


CREATE TRIGGER tri 6 

ON readinfo 

INSTEAD OF delete 

BEGIN 

DECLARE Q@bookid int; 

SELECT @bookid=bookid FROM deleted; 

UPDATE bookinfo SET bookcount=bookcount+1 WHERE bookid=@bookid; 


-馆藏 图 书 数量 加 


END 


执行 上 面 的 语句 ， 效 果 如 图 13.12 所 示 。 
完成 触发 器 tri_6 的 创建 后 ， 下 面 就 来 验证 其 效果 了 。 验 证 的 语句 如 下 : 


SELECT * FROM bookinfo; -查询 图 书信 息 表 的 原始 数据 
SELECT * FROM readinfo; -查询 借阅 信息 表 的 原始 数据 
DELETE FROM readinfo WHERE userid=2 AND bookid=1003; 

SELECT * FROM readinfo; 一- 查询 执行 删除 语句 后 的 借阅 信息 表 
SELECT * FROM bookinfo; 一 -查询 还 书后 的 图 书信 息 表 


执行 上 面 的 语句 ， 效 果 如 图 13.13 所 示 。 
从 图 13.13 所 示 的 效果 可 以 看 出 ，INSTEAD OF 触发 器 确实 是 阻止 了 数据 表 执 行 删除 
操作 。 
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Ee Microsor: sotservervanaoemcntstuio 
文件 编辑 (E) ”视图 四， 查询 ) 项 目 (E) 调式 D) 工具 (D 窗口 ( 拉 社区 (QO 帮助 中 
:了 | 记 | 也 也 驴 | 记 | 号 是 马 | 划 电 
: My a | dapterl3 -| ?1 执 和 WW 》 昌 v 品名 国政 模 | 给 圈 避 
“SQLQuery1.sql .istrator (52))*| vx 
章 CREATE TRIGGER tri 6 

ON readinfo 

INSTEAD OF delete 

35 

BEGIN 

DECLARE Bbookid int; 

SELECT Bbookid=bookid FRON deleted; 

UPDATE bookinfo SET bookcount=bookcount+1 WHERE bookid=Bbookid; 

END 了 


园 查 .| wipw-9sz7WMS5QLsERVER2008 (… | WIDW-9527VAdministrato,,，| chapter13 | 00:00:00 | 0 行 
就 绪 行 13 列 1 Chl Ims 大 


图 13.12 创建 触发 器 tri 6 
SIE 


文件 日 ”编辑 (E) 视图 查询 Q) 项 目 (P) 调试 D) 工具 (D 窗口 Ww) 社区 (QO 帮助 () 
卫生 | 记 | 动 孙 马 | 记 | 荡 回忆 | 骂 蝇 


By ba | donewls "1! a v 33 昌国 | 守 码 | 好 图 
SQLQuery1.sql -istrator (52))* | vx 


站 数据 a 


日 SELECT * FROM bookinfo; ”-- 查 词 图 书信 息 表 的 原 
SELECT * FROM readinfo;  -- 查 询 借阅 信 
DELETE FROMK readinfo WHERE userid=2 
SELECT * FRON readinfo; 一 二 人 天 

SELECT * FROM bookinfo: ”-- 查 询 还 书后 的 图 书信 | 


BE rt 和 | chapter13 | 00:00:00 | 5 行 
就 绪 第 1 行 第 ! 列 4 


图 13.13 ”验证 触发 器 tri_6 的 效果 


13.2.4 创建 带 加 密 选 项 的 触发 器 


所 谓 带 加 密 选 项 就 像 在 上 一 章 中 存储 过 程 使 用 的 加 密 选 项 一 样 ， 只 要 在 创建 触发 器 时 
为 其 加 上 WITH ENCRYPTION 就 可 以 为 创建 触发 器 的 文本 加 密 了 。 下 面 就 使 用 示例 7 来 
演示 如 何 创建 带 加 密 选 项 的 触发 器 。 

【示例 7】 创建 触发 器 tri 7， 并 使 用 加 密 选项 设置 。 触 发 器 的 作用 是 禁止 借阅 价格 高 
于 30 元 的 图 书 。 
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根据 题目 要 求 ， 使 用 的 是 AFTER 类 型 的 触发 器 。 具 体 的 创建 语句 如 下 : 


CREATE TRIGGER tri 7 
ON readinfo 


WITH ENCRYPTION -- 加 密 文本 

AFTER insert 

RS 

BEGIN 

DECLRARE Q@bookid int,@price decimal (6,2); 

SELECT @bookid=bookid FROM inserted; -- 获 取 添 加 的 图 书 编号 

if (EXISTS (SELECT * FROM bookinfo WHERE bookid=@bookid AND bookprice>30) ) 

BEGIN 

ROLLBACK TRANSACTION; -- 回 滚 事务 

END 

了 ELSE 

BEGIN 

UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=@bookid; 
-- 更 新 图 书 数量 

END 

END 


执行 上 面 的 语句 ， 效 果 如 图 13.14 所 示 。 
验证 触发 器 tri_7 是 否 被 加 密 了 ， 仍 然 可 以 使 用 系统 存储 过 程 SP_HELPTEXT 来 查看 。 
具体 的 查看 语句 如 下 : 


SP_HELPTEXT tri 77 


执行 上 面 的 语句 ， 效 果 如 图 13.15 所 示 。 


RMicrosoft SQL Server Management Studio =I9jx| 
文件 中 ” 蝙 鱼 5) 视图) 查 注 0) 项 目 B) 调 #(D) 工具 D 窗 D(w 区 (CO) 帮助 td 

也 订 建 查询 | 由 轧 也 区 记忆 中 号 马 二 

dew1s "17 执 50 有 Y 弛 屯 国 江 艳 | 外 图 吧 | 三 全 | 幸 闻 


SQLQuery1.sql .istrator (52))* | 3 x| 
站 CREATE TRIGGER tei_7 一 
ON readinfo 
VITH ENCRYPTION 一 加 密 文本 
APTER insert 
所 
BEGIN 
DECLARE Bbookid ine,Bprice dee 
SELECT Qhookid-bookid FROM 。 -的 图 忆 入 号 ， =I 
4£ (EXISTS SELECT * FRON boo WHERE bookid"Bbo cice>30 文件 (篇 加 (视图 VW) 查询 Q) 项目) 
BEGIN 调 0) 工具 D 窗 D(Ww) 社区 (C。 帮助 
ODE TRANSACTION; -回流 事务 巡 吉 可) 六 六 迎 访 | 忆 | 已 吕 
ELSE Wy 28 | chapter13 - 总 
BEGIN 二 
UPDATE bookinto SET bookcount=bookcounc-1 WHERE bookid-Bbookid; -- 更 新 图 书 数量 SQLQueryl.sql -istrator (52)) -x 
日 SP_HELPTEXT tri_7 ~- 
| | 
«ll 
| 
对 象 “cri_7， 的 文本 已 加 密 。 = 
可 » 
加 查询 已 成 功 执行 。 | WIDW-9527\M55QLSERVER2008 (... WIDW-9527\Admnistrato..，| chapter13 | 00:00:00 | 0 行 Pw-9527\Administrato..，| chapter13 | 00:00:00 | 0 行 
就 绪 行 19 列 1 chi Ins /4 就 绪 行 叫 1chi 4 
图 13.14 ”创建 带 加 密 选项 的 触发 器 tri 7 图 13.15 查看 触发 器 的 创建 文本 


通过 图 13.15 所 示 的 效果 可 以 看 出 , 通过 WITH ENCRYPTION 确实 是 将 创建 触发 器 的 
文本 加 密 了 。 


全 注意 : 在 创建 触发 器 时 , 读者 要 注意 的 是 每 张 表 的 操作 ( INSERT、 UPDATE 和 DELETE ) 
只 能 有 一 种 类 型 的 触发 器 。 
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13.3 ”修改 触发 器 


触发 器 在 创建 完成 后 ， 有 时 还 是 要 对 原 有 的 触发 器 进行 修改 的 。 修 改 触发 嚣 不仅 可 以 
修改 触发 器 中 的 语法 ， 也 可 以 通过 语句 来 设置 触发 器 是 否 禁 用 。 在 本 节 中 就 带领 读者 一 同 
来 学 习 如 何 使 用 SQL 语句 修改 触发 器 。 在 后 面 的 13.5 节 中 还 将 讲解 如 何 使 用 企业 管理 器 


来 修改 触发 器 。 
13.3.1 修改 触发 器 的 语法 


修改 触发 器 的 语法 与 创建 触发 器 的 语法 类 似 ， 只 是 将 CREATE 关键 字 换 成 了 ALTER 
关键 字 。 有 具体 语法 形式 如 下 : 


ALTER TRIGGER trigger name 

ON { table | view } 

[ WITH ENCRYPTION] 

{ FOR | AFTER | INSTEAD OF } 
TNSPREOTN 1 UPDATE TD DELETE 
[ NOT FOR REPLICATION ] 

AS { sql statement } 


其 中 : 


口 
口 
口 
口 
口 


口 


口 
口 


trigger_name: 触发 器 的 名 称 。 

table | view: 触发 器 作用 的 表 名 或 视图 名 。 

WITH ENCRYPTION: 对 文本 进行 加 密 。 与 它 在 存储 过 程 中 的 含义 一 样 。 
FORIAFTER: 当 执 行 某 些 操作 后 被 激发 。 比 如 : 向 表 中 添加 数据 后 激发 。FOR 与 
AFTER 是 同 义 的 。 

INSTEAD OF: 蔡 代 操 作 ， 需 要 注意 的 是 对 于 表 或 视图 ， 每 个 INSERT、UPDATE 
或 DELETE 语句 最 多 可 定义 一 个 INSTEAD OF 触发 器 。 

{[DELETE] [,] [INSERT] [,] [UPDATE] }: 指定 在 哪 种 操作 时 激发 触发 器 。 可 以 选 
择 1 到 多 个 选项 。 

NOT FOR REPLICATION: 当 复 制 表 时 ， 触 发 器 不 被 激发 。 

sql_statement: 触发 器 被 激发 时 执行 的 工 SQL 语句 。 


13.3.2 ”修改 触发 器 
有 了 修改 触发 器 的 语法 ， 就 可 以 尝试 修改 触发 器 了 。 下 面 就 通过 示例 8 来 讲解 如 何 修 


改 触发 器 。 
【示例 8】 修改 在 示例 7 中 创建 的 触发 器 tri_7， 改 成 当 出 版 社 是 机 械 工业 出 版 社 时 图 
书 禁止 借阅 。 


根据 题目 要 求 ， 修 改 的 语句 如 下 : 


ALTER TRIGGER tri 7 

ON readinfo 

WITH ENCRYPTION =-- 加 密 文本 
AFTER insert 
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AS 
BEGIN 
DECLARE @bookid int,@price decimal (6,2); 
SELECT @bookid=bookid FROM inserted; -获取 添加 的 图 书 编号 
if (EXISTS (SELECT * FROM bookinfo WHERE bookid=@bookid AND bookpub=' 机 械 工 
la) 
BEGIN 
ROLLBACK TRANSACTION; -- 回 滚 事务 
END 
ELSE 
BEGIN 
UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=@bookid; 
-- 更 新 图 书 数量 
END 
END 
执行 上 面 的 语句 ， 效 果 如 图 13.16 所 示 。 
-15x 


文件 (编辑 (E) ” 视 苞 (Y) 查询 (9) 项 目 (E) 调试 D) 工具 (D 窗口 (Ww) 社区 (CQ) 帮助 (中 
2 半 坦 WY | 广 | 疗 池 怨 | 访 | 蕊 回 当 | 台 忆 
YW | chapter13 -| ?执行 0 下 v 品 曲 回 | 3” 绚 | 的 圈 太 | 三 全 | 素 率 
SQLQuerylsql istrator (52))*| vxX 
ALTER TRIGGER tri 7 
ON readinfo 
WITH ENCRYPTION -- 加 密 文本 
AFTER insert 
AS 
BEGIN 
DECLARE Bbookid int,Bprice decimal (6,2); 


SELECT Bhookid=bookid FRONM inserted; -- 获 取 添 加 的 图 书 编号 

if (EXISTS (SELECT * FRONM bookinfo WHERE bookid=Bbookid AND bookpub=' 机 械 工 业 ')) 
BEGIN 

ROLLBACK TRANSACTION; -- 回 注 事 务 

END 

ELSE 

BEGIN 


UPDATE bookinfo SET bookcount=bookcount-1 WHERE bookid=Bbookid; -- 更 新 图 书 数量 
END 
上 END 


己 消息 | 


命令 已 成 功 完成 。 
加 此 zp 
多 查询 已 成 功 执行 。 | wIpW-9527WMS5QLSERYER2008 (,，| WIDW-9527WAdministrato,,，| chapter13 | 00:00:00 | 0 行 


就 绪 行 18 


图 13.16 ”修改 触发 器 tri_7 


从 图 13.16 所 示 的 界面 ， 可 以 看 出 触发 器 的 内 容 已 经 被 修改 了 。 那 么 ， 现 在 就 请 读者 
编写 验证 语句 ， 自 己 来 验证 一 下 修改 后 的 触发 器 tri_7 吧 。 


13.3.3 ”禁用 /启用 触发 器 


通过 前 面 讲解 的 修改 触发 器 的 语法 是 不 能 够 将 触发 器 禁用 的 。 禁 用 触发 器 的 目的 是 当 
一 些 触发 器 目前 不 用 ， 但 是 以 后 可 能 还 会 使 用 时 ， 不 需 直 接 将 触发 器 删除 。 如 果 今 后 想 继 
续 使 用 被 禁用 的 触发 器 ， 直 接 使 用 语句 将 其 启用 就 可 以 了 。 下 面 就 分 别 来 讲解 如 何 禁用 和 
启用 触发 器 。 

1. 禁用 触发 器 


用 触发 器 是 通过 DISABLE 语句 来 完成 的 。 具 体 的 语法 结构 如 下 : 


列 1 Chl ms 


刍 
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DISABLE TRIGGER {[trigger name [,...n ] | ALL } 
ON object name 
其 中 : 
口 trigger_name: 触发 器 的 名 称 。 
口 ALL: 所 有 触发 器 。 
口 object_name: 要 禁用 触发 器 的 表 或 视图 。 
下 面 就 通过 示例 9 来 练习 如 何 禁 用 触发 器 。 
【示例 9】 禁用 触发 器 tri_7。 
使 用 DISABLE TRIGGER 语句 禁用 触发 器 tri_7， 触 发 器 tri_7 是 作用 在 借阅 信息 表 
(readinfo) 上 的 。 具 体 的 语句 如 下 : 


DISABLE TRIGGER tri 7 
ON readinfo; 


执行 上 面 的 语句 ， 效 果 如 图 13.17 所 示 。 TT 


文件 编辑 (E) 视力 
通过 图 13.17 所 示 的 效果 , 就 说 明 禁 用 触发 器 的 操作 已 经 。 于 昌 于 区 
完成 了 。tri_7 触发 器 是 用 来 借阅 图 书 时 的 判断 ， 那 么 ， 读 者 | 汉 aesau| 记 | 孔子 号外 与 


时 Ha | chapter13 


可 以 试 着 编写 一 条 借阅 图 书 的 语句 来 看 看 触发 器 是 否 还 起 作 [RCR x 


用 呢 ? ne 
2. 启用 触发 器 ET 
ls ,于 
当 触 发 器 需要 重新 恢复 其 作用 时 , 就 需要 重新 启用 该 触发 。 ERROR oooomo 0 
器 了 。 启 用 触发 器 使 用 的 是 ENABLE 语句 。 具 体 的 语法 形式 时 mo4 
如 下 : 图 13.17 禁用 触发 器 tri_7 


ENABLE TRIGGER { [trigger_name [,...n ] | ALL } 

ON object name 

其 中 : 

口 trigger_name: 触发 器 的 名 称 。 

口 ALL: 所 有 触发 器 。 

口 object name: 要 禁用 的 触发 器 的 表 或 视图 。 

实际 上 ， 读 者 会 发 现 启用 和 禁用 触发 器 的 语法 非常 类 似 ， 千 万 不 要 混 消 啊 ! 下 面 就 通 
过 示例 10 来 演示 启用 触发 器 。 

【示例 10】 启用 触发 器 tri_7。 

使 用 ENABLE TRIGGER 语句 启用 触发 器 tri_7， 触 发 器 tri 7 是 作用 在 借阅 信息 表 
(readinfo) 上 的 。 具 体 的 语句 如 下 : 

ENABLE TRIGGER tri 7 

ON readinfo; 

执行 上 面 的 语句 ， 效 果 如 图 13.18 所 示 。 

通过 图 13.18 所 示 的 效果 ， 就 说 明 启用 触发 器 的 操作 已 经 完成 了 。 也 就 是 说 ， 触 发 器 
tri 7 又 恢复 了 原来 的 功能 。 不 信 可 以 试 试 哦 ! 
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项 目 (E) 调式 D) 工具 D 窗 DWw) 
社区 (OQ 各 助 4 
也 霜 汗 可 | 让 已 孙 下放 中 


,rr 
日 PNABLE TRIGGER tri 7 


图 13.18 ”启用 触发 器 tri_7 


13.4 删除 触发 器 


当 某 些 触发 器 以 后 都 不 再 使 用 时 ， 可 以 考虑 删除 触发 器 。 删 除 后 的 触发 器 是 不 能 够 恢 
复 的 。 因 此 ， 在 考虑 要 删除 某 些 触发 器 时 ， 一 定 要 慎重 ! 如 果 不 能 确定 触发 器 是 否 不 用 ， 
可 以 先 将 这 些 触发 器 做 禁用 操作 。 

下 面 就 先 来 看 看 删除 触发 器 的 语法 吧 , 删除 触发 器 仍然 使 用 的 是 DROP 语句 来 完成 的 。 
具体 的 语法 形式 如 下 : 


DROP TRIGGER trigger name [ ,...n] [;}] 


这 里 ，trigger_name 是 指 要 删除 的 触发 器 的 名 称 。 在 删除 触发 器 时 一 次 可 以 删除 一 个 
或 多 个 触发 器 。 多 个 触发 器 之 间 用 去 号 隔 开 即 可 。 

下 面 就 来 演示 使 用 DROP TRIGGER 语句 删除 触发 器 。 

【示例 11】 删除 触发 器 tri_7。 

使 用 DROP TRIGGER 语句 删除 tri_7， 具 体 的 语句 如 下 : 


DROP TRIGGER tri_7; 


执行 上 面 的 语句 ， 效 果 如 图 13.19 所 示 。 


gly 
文件 (E) ”编辑 (E) ”视图 (Y) ”查询 (9) 
项 目 (p) 调式 D) 工具 (DD 窗口 (W) 


社区 (QO 帮助 (中 ) 
也 站 查询 | 让 | 孔 辽 台 | 记 上 
BY 2 | chapterl3 外 
‘SQL Queryl.sql -istrator (52))* vx 
章 DROP TRIGGER tri 7; — 
I 
已 消息 | 
命令 已 成 功 完成 。 
可 
S27\Administrato... | chapter13 | 00:00:00 | 0 行 
就 绪 和 th 2 


图 13.19 ”删除 触发 器 tri_7 
这 样 ， 在 数据 库 chapter13 中 就 没有 名 为 tri_7 的 触发 器 了 。 
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13.5 ”使 用 企业 管理 器 管理 触发 器 


在 前 面 的 内 容 中 全 部 都 是 通过 SQL 来 操作 触发 器 的 , 有些 读者 学 到 这 里 就 会 感到 有 些 
疲惫 了 ， 但 是 ， 不 要 紧 ， 本 节 的 内 容 就 要 轻松 多 了 。 在 企业 管理 器 中 操作 触发 器 就 免 去 了 
记 语 法 的 工作 了 ， 直 接 按照 步骤 操作 就 行 了 。 有 了 前 面 使 用 企业 管理 器 操作 存储 过 程 的 基 
础 ， 再 使 用 其 操作 触发 器 ， 这 对 于 读者 来 说 就 是 小 菜 一 碟 了 。 


13.5.1 使 用 企业 管理 器 创建 触发 器 


在 企业 管理 器 中 创建 触发 器 是 非常 容易 的 ， 通 常 需要 两 个 步 又 就 可 以 完成 了 。 下 面 就 
使 用 示例 12 来 演示 如 何在 企业 管理 器 中 创建 触发 器 。 

【示例 12】 创建 触发 器 tri_8， 当 删除 读者 信息 表 的 读者 信息 时 ， 需 要 将 读者 的 借阅 信 
息 也 一 并 删除 。 

使 用 企业 管理 器 完成 创建 触发 器 tri_8 的 工作 ， 有 具体 分 为 如 下 两 个 步骤 。 

(1) 在 “对 象 资源 管理 器 ”中 ， 在 数据 库 目 录 下 ， 找 到 chapter13， 并 在 表 目 录 下 找到 
读者 信息 表 〈userinfo)， 展 开 表 目录 ， 右 击 “ 触 发 器 ”节点 ， 在 弹出 的 右键 菜单 中 选择 “新 
建 触 发 器 ”选项 ， 如 图 13.20 所 示 。 


a Microsoft SQL Server Man Studio 
文件 日。 编辑 (E) 视图 (四 项 目 (EB) 调式 D) 工具 (D 窗口 (W) 社区 (QO 帮助 (H) 
PE EE 


Ry Wa | dhepterl -1 执 5W 》 sv 加 名 国政 的 | 公 圈 们 | 三 | 来 素 | 包 局 
时光 四 so Query3.sql -nistrator (52))| -x 
连接 oO)" | 对 到 了 马 SET ANSI_NULLS ON 习 


日 图 WIDw-9527\M55QLSERVER20C a 
名 SET QUOTED_IDENTIFIER ON 


ne> .<Trigger_Name, sysname, Trigger_ Nae; 
Teble_ Name, sysname, Table_Name 
INSERT, DELETE, UPDATE> 


-- SET NOCOUNT ON added to prevent extra result sets from 
-- interfering with SELECT statements. 
SET NocOUNT oN 


-- Insert statements for trigger here 


= 
4 


上 
都 已 连接 . (1/1) | WIDW-9527\MS5QLSERVER2008 {... | WIDW-9527\Administrato...| chapter13 | 00:00:00 |0 行 
就 绪 行 1 列 1 chl Ins /4 


图 13.20 创建 触发 器 界面 


(2) 在 图 13.20 所 示 界 面 中 ， 根 据 需 要 修改 相应 的 参数 ， 具 体 语 句 如 图 13.21 所 示 。 
添加 完成 后 ， 再 单 击 “ 执 行 ”按钮 ， 即 可 完成 触发 器 的 创建 操作 。 

至 此 ， 触 发 器 tri_8 就 创建 完成 了 。 

除了 像 示 例 12 中 讲解 的 直接 在 创建 触发 器 的 界面 填 入 相应 的 参数 外 , 也 可 以 通过 “ 指 
定 模板 参数 的 值 ”界面 来 完成 参数 的 添加 。 具 体 的 方法 是 单 击 企 业 管理 器 菜单 栏 中 的 “ 查 
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crocof so Sever Monogement Studio 
文件 E) ”编辑 (视图 W) 查询 (Q) 项 目 (EC) 调 江 (0) 工具 (D 窗口 Ww) 
社区 (CO) 帮助 由 

了 新 寻 查 芭 | 让 | 轧 也 怠 记 | 蕊 加 本 | 怠 瑟 

服 W | cheptert3 | 
“SQLQuery3.sql -istrator (52))*| 2 


Le ee pn 


SET ANSI_NULLS ON 
Go 


SET QUOTED_IDENTIFIER ON 
Go 


eate date 
-- Description 


日 CREATE TRIG 
ON userinfo 
AFTER DELETE 


-— Insert statements for trigger here 
DECLARE Buserid int; 

SELECT Buserid=userid FRON deleted; 
DELETE 了 RON readinfo WHERE userid=Buserid; 


1.9527WM55QLSERVER2008 (... | WIDW-9527WAdministrato,,，| chapter13 | 00:00:00 | 0 行 
就 绪 行 3 列 1 chl 


图 13.21 创建 触发 器 tri_8 
询 ”|“ 指 定 模 板 参 数 的 值 ” 选 项 ， 出 现 图 13.22 所 示 界 面 。 


Create Date 


Description 


Schena_Nane sysnane Schena_ Nane 
Trigger_Nane sysnane Trigger_Nane 
Table_Nane sysnane Table_ Nane 

Data Modification St INSERT, DELETE, UPDATE 


Cw |] ww | ww | 


4 


图 13.22 ”指定 模板 参数 的 值 


在 图 13.22 所 示 的 界面 中 , 填 入 相应 的 参数 值 , 并 在 触发 器 创建 界面 中 填 入 相应 的 SQL 
语句 即 可 。 


13.5.2 ”使 用 企业 管理 器 修改 触发 器 


在 企业 管理 器 中 修改 触发 器 要 比 创建 触发 器 的 操作 容易 一 些 ， 下 面 就 简单 地 讲解 如 何 
在 企业 管理 器 中 修改 触发 器 。 实 际 上 ， 修 改 触发 器 时 也 只 需要 两 个 步骤 即 可 。 这 里 仍 以 修 
改 tni_8 为 例 。 
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(1) 在 “对 象 资源 管理 器 ”中 ， 在 数据 库 目 录 下 ， 找 到 chapter13， 并 在 表 目 录 下 找到 
读者 信息 表 (userinfo)， 展 开 表 目 录 ， 展 开 “ 触 发 器 ”节点 ， 并 右 击 tri_8 触发 器 。 在 弹出 
的 右键 菜单 中 选择 “修改 ”选项 ， 如 图 13.23 所 示 。 


文体 器 ” 编 各 @ 视图 查询 O) 项 目 B) 调式 D) 工具 D 窗口) 社区 (C) 和 助 中 
;也 新建 查 询 W | 记 | 了 也 可 | 让 | 区 加 己 | 受 了 
-36o 》 = v 多 名 加 | 玉 二 | 的 图 入 | 三 EA 


“5QLQuery5.sql -istrator (55))* | SQLQUery4.sq ~.. nistrator (53)) T zx 
USE [chapter13] 
GO 
/eee Object: Trigger [dbo].[tri 8] Script D 
SET ANSI_NULLS ON 
GO 
辐 hapterl0 SET QUOTED_IDENTIFIER ON 
田 国 chapterll 
田 国 chapterl2 9 
日 转 dhapterl3 -- Author: <Author, 
田 国 数据 库 关系 图 - Create date: <Create Date, 
二 = ee <Description,,> 


日 ALTER TRIGGER [dbo].[tri_8] 
ON [dbo].[userinfo] 
AFTER DELETE 
AS 
BEGIN 
-- Insert statements for trigger here 
DECLARE Buserid int; 
SELECT Buserid=userid FROM deleted; 
DELETE FRON readinfo WHERE userid=Buserid; 


4 
| 27WM55QLSERYER2008 (,.,| WIDW-9527WAdministrato,,，| chapter13 | 00:00:00 | 0 行 
行 24 列 1 Chl Ins 


图 13.23 ”修改 触发 器 界面 


(2) 在 图 13.23 所 示 界 面 中 , 按照 要 求 修改 相应 的 参数 和 SQL 语句 , 完成 修改 操作 后 ， 
直接 单 击 “ 执 行 ”按钮 ， 即 可 完成 触发 器 的 修改 操作 。 


13.5.3 ”使 用 企业 管理 器 删除 触发 器 


删除 触发 器 可 谓 是 最 简单 的 一 个 操作 了 ， 但 是 ， 这 个 操作 也 是 最 危险 的 ， 删 除 后 的 触 
发 器 就 不 能 恢复 了 。 下 面 就 通过 示例 13 来 讲解 如 何 删除 触发 器 tri_8。 

【示例 13】 删除 触发 器 tri 8。 
用 语句 删除 触发 器 就 是 一 句 话 的 事 , 相对 于 语句 来 说 ,使 用 企业 管理 器 就 略 显 麻烦 了 。 
下 面 就 来 看 看 具体 的 操作 步骤 。 

(1) 在 “对 象 资源 管理 器 ”中 ， 在 数据 库 目 录 下， 找到 数据 库 chapter13， 并 找到 要 删 
除 触 发 器 所 在 的 数据 表 userinfo。 展 开 其 下 的 触发 器 节点 ， 右 击 tri_8 触发 器 ， 在 弹出 的 右 
键 菜单 中 选择 “删除 ”命令 。 弹 出 图 13.24 所 示 界 面 。 

(2) 在 图 13.24 所 示 界 面 中 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 删除 触发 器 tri_8 的 操作 。 


13.5.4 使 用 企业 管理 器 启用 /禁用 触发 器 


在 企业 管理 器 中 启用 或 者 禁用 触发 器 的 操作 都 与 删除 触发 器 的 操作 有 些 类 似 ， 也 是 非 
常 简 单 的 。 下 面 就 分 别 来 讲解 如 何 启用 和 禁用 触发 器 。 
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=IGl 光 | 


服务 器 
rr 


连接 
WIDN-9527 Wdninistrater 


扩展 考 
EE 
ET 


|) 
图 13.24 ”删除 提示 界面 


1. 禁用 触发 器 


禁用 触发 器 的 操作 分 为 如 下 两 个 步骤 。 这 里 以 禁用 tri_8 为 例 。 

(1) 在 “对 象 资源 管理 器 ”中 ， 在 数据 库 目录 下 ， 找 到 数据 库 chapter13， 并 找到 要 删 
除 触 发 器 所 在 的 数据 表 userinfo。 展 开 其 下 的 触发 器 节点 ， 右 击 tri_8 触发 器 ， 在 弹出 的 右 
键 菜单 中 选择 “禁用 ”命令 。 弹 出 图 13.25 所 示 界 面 。 


而 茶 用 捉 发 器 - WIDW-9527\MSSQLSERYER2006 -lolx| 
© 1 总 计 0 错误 
4 1 成 功 0 要 
详细 信息 中) : 


禁用 触发 器 “tri_8” 


图 13.25 ”禁用 触发 器 界面 
(2) 在 图 13.25 所 示 的 界面 中 ， 单 击 “ 关 闭 ” 按 钮 ， 即 可 完成 触发 器 的 禁用 操作 。 
2. 启用 触发 器 


有 了 禁用 触发 器 的 基础 ， 启 用 触发 器 的 操作 就 变 得 更 加 容易 了 。 启 用 tri_8 触发 器 ,分 
为 如 下 两 个 步骤 。 
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(1) 在 “对 象 资源 管理 器 ”中 ， 在 数据 库 目 录 下 ， 找 到 数据 库 chapter13， 并 找到 要 删 
除 触 发 器 所 在 的 数据 表 userinfo。 展 开 其 下 的 触发 器 节点 ， 右 击 tri_8 触发 器 ， 在 弹出 的 右 
键 菜单 中 选择 “启用 ”命令 。 弹 出 图 13.26 所 示 界 面 。 


图 启用 触发 器 - WIDW-9527\MSSQLSERYER2008 -Iolx| 
1 总 计 0 钳 误 
CE 1 三 功 0 可 省 


详细 信息 中) : 


图 13.26 启用 触发 器 界面 
(2) 在 图 13.26 所 示 的 界面 中 ， 单 击 “ 关 闭 ” 按 钮 ， 即 可 完成 触发 器 的 启用 操作 。 


13.6 本 章 小 结 


在 本 章 中 主要 讲解 了 触发 器 的 作用 、 分 类 以 及 如 何 创建 、 修 改 和 删除 DML 类 型 触发 
器 。 在 创建 触发 器 时 讲解 了 创建 AFTER 、INSTEAD OF 以 及 带 加 密 选 项 的 触发 器 ; 在 修改 
触发 器 中 除了 讲解 了 使 用 一 般 语 法 修改 触发 器 的 操作 , 还 介绍 了 禁用 和 启用 触发 器 。 此 外 ， 
还 讲解 了 如 何 使 用 企业 管理 器 来 创建 、 修 改 以 及 删除 触发 器 。 相 信 通 过 本 章 的 学 习 ， 读 者 
能 够 更 好 地 理解 触发 器 的 功能 以 及 如 何 使 用 触发 器 。 在 以 后 的 工作 中 ， 有 选择 地 使 用 触发 
器 必 能 起 到 事半功倍 的 效果 ! 


13.7 本 章 习 题 


一 、 填 空 题 
1. 在 SQL Server 中 ，DML 类 型 触发 器 的 操作 指 的 是 操作 。 
2. 在 触发 器 中 有 两 个 特殊 的 表 : 和 8 
3. INSTEAD OF 触发 器 的 作用 是 
二 、 选 择 题 
1. 下 列 哪个 是 创建 触发 器 的 语句 。 
A. CREATE PRO B. CREATE TRIGGER 
C. ALTER TRIGGER D. 以 上 都 不 对 
2. 下 列 哪个 是 禁用 触发 器 的 语句 R 
A. USE B. DISABLE C. ENABLE D. 以 上 都 不 对 
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3. 下 列 哪个 是 删除 触发 器 的 语句 


A. DELETE TRIGGER 


C. DROP TRIGGER 
三 、 问 答题 


1. 触发 器 有 哪儿 种 类 型 ? 


B. CLOSE TRIGGER 
D. 以 上 都 不 对 


2. 使 用 触发 器 时 需要 注意 什么 问题 ? 
3. DML 触发 器 中 ，AFTER 类 型 触发 器 和 INSTEAD OF 类 型 触发 器 有 什么 区 别 ? 


创建 用 户 信 息 表 和 日 志 信息 表 ， 表 结构 如 表 13-6 和 表 13-7 所 示 。 并 完成 下 列 触发 器 


表 13-6 用 户 信 息 表 (userinfo) 


四 、 操 作 是 

作 。 

序号 列 名 
1 id 
2 username 
3 userpwd 

序号 列 名 
1 id 
2 userid 
3 username 
4 userpwd 


数据 类 型 
int 
varchar(20) 
varchar(20) 


数据 类 型 
int 

int 
varchar(20) 


varchar(20) 


说 明 
用 户 编号 
用 户 名 
密码 


说 明 
日 志 编号 
用 户 编导 
用 户 名 
密码 


(1) 创建 触发 器 tri_1， 完 成 当 向 用 户 信息 表 插入 数据 时 ， 同 时 向 用 户 信息 日 志 表 插入 


的 数据 。 


(2) 创建 触发 器 tri_2， 完 成 当 删 除 用 户 信息 表 数据 时 ， 同 时 向 用 户 信息 日 志 表 插 入 删 
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第 14 章 与 数据 安全 相关 的 对 象 


确保 数据 库 中 数据 的 安全 性 是 每 一 个 从 事 数据 库 管 理工 作 人 员 的 理想 。 但是， 无 论 什 
么 样 的 数据 库 设 计 都 不 能 说 是 十 全 十 美的 ， 只 是 说 尽量 地 提高 数据 库 的 安全 性 。 那 么 ， 如 
何 提高 数据 库 的 安全 性 呢 ? 这 就 与 本 章 要 学 习 的 数据 库 对 象 息息相关 叶 ! 

章 的 主要 知识 点 如 下 : 

口 了 解 与 数据 安全 相关 的 对 象 

口 如 何 管理 用 户 

口 如 何 管理 角色 

口 如 何 管理 权限 

口 如 何 管理 登录 账号 


14.1 认识 与 数据 安全 相关 的 对 象 


在 SQL Server 数据 库 中 ， 能 够 对 数据 安全 起 作用 的 对 象 主要 有 用 户 、 权 限 、 角 色 以 及 
登录 账号 。 只 有 了 解 了 这 些 对 象 的 作用 ， 才 能 够 灵活 地 设置 和 使 用 这 些 对 象 。 在 本 节 中 将 
带领 读者 一 一 与 其 见面 。 

(1) 数据 库 用 户 

所 谓 数据 库 用 户 ， 就 是 指 能 够 使 用 数据 库 的 用 户 。 在 SQL Server 中 ， 可 以 为 不 同 的 数 
据 库 设置 不 同 的 用 户 ， 从 而 提高 数据 库 访 问 的 安全 性 。 

在 SQL Server 数据 库 中 有 两 个 特殊 的 用 户 ， 一 个 是 dbo 用 户 ， 一 个 是 guest 用 户 。 这 
两 个 用 户 之 所 以 特殊 ， 是 因为 安装 系统 后 就 存在 了 ， 并 且 它 们 默认 就 存在 于 每 一 个 数据 库 
中 。guset 用 户 的 特点 是 可 以 被 禁用 的 ; dbo 用 户 的 特点 是 创建 数据 库 对 象 的 所 有 者 默认 为 
都 是 dbo 用 户 ， 并 且 该 用 户 是 不 能 删除 的 。 

(2) 用 户 权 限 

了 解 了 数据 库 用 户 ， 那 么 ， 用 户 究 竞 能 如 何 操作 数据 库 呢 ?这 个 还 是 可 以 由 数据 库 管 
理 员 来 指定 的 ， 指 定 用 户 做 什么 就 靠 给 用 户 设置 权限 来 完成 的 。 比 如 : 在 商场 中 经 常会 7 
理 各 种 会 员 卡 ， 每 种 会 员 卡 一 般 会 有 普通 会 员 、VIP 会 员 ， 那 么 每 种 卡 持 有 的 会 员 在 消费 
时 都 有 不 同 的 折扣 。 用 户 也 是 一 样 的 ， 通 过 对 用 户 设置 权限 ， 每 个 数据 库 用 户 都 会 有 不 同 
的 访问 权限 ， 比 如 : 让 用 户 只 能 查询 数据 库 中 的 信息 不 能 更 新 数据 库 的 信息 。 

(3) 角色 
色 这 个 词 ， 读 者 应 该 不 会 太 陌生 。 在 电影 或 电视 剧 中 经 常会 提 到 角色 ， 比 如 : 主演 、 
友情 出 演 等 等 。 那 么 ， 如 果 你 有 幸 成 为 某 电 影 的 主演 ， 在 这 个 电影 中 所 承担 的 任务 按照 剧 
本 就 已 经 明确 了 ; 同样 ， 如 果 你 是 友情 出 演 某 个 电影 ， 也 会 在 剧本 中 有 确定 的 任务 ， 只 是 
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任务 不 多 罢了 。 在 数据 库 中 也 是 一 样 的 ， 角 色 是 在 数据 库 中 设置 好 权限 的 ， 合 理 地 分 配给 
用 户 就 可 以 了 。 通 常 ， 将 角色 看 作 是 一 些 权 限 的 集合 。 因 此 ， 如 果 需 要 给 用 户 设置 很 多 权 
限时 ， 能 够 直接 找到 适合 的 角色 ， 就 可 以 将 设置 权限 的 工作 变 得 容易 多 了 。 那 么 ， 在 数据 
库 中 究竟 有 哪些 角色 呢 ? 在 SQL Server 中 ， 角 色 可 以 分 为 3 种 , 分别 是 数据 库 角 色 、 服 务 
器 角色 以 及 应 用 程序 角色 。 至 于 每 种 角色 中 都 包含 哪些 权限 ， 将 在 本 章 的 角色 部 分 详细 
讲解 。 

(4) 登录 账号 

登录 账号 是 用 来 访问 SQL Server 数 据 库 系统 使 用 的 。 它 不 同 于 前 面 所 讲 的 数据 库 用 户 。 
户 是 用 来 访问 某 个 特定 数据 库 的 。 一 个 登录 账号 可 以 访问 多 个 数据 库 ， 而 一 个 用 户 只 能 
访问 特定 的 数据 库 ， 并 且 不 能 直接 访问 SQL Server 系统 ， 只 有 给 用 户 设置 登录 账号 映射 才 
能 访问 SQL Server 系统 。 因 此 ， 合 理 地 控制 用 户 使 用 登录 账号 ， 也 是 确保 数据 库 安全 性 的 
一 个 手段 。 


14.2 ”登录 账号 管理 


所 谓 登 录 账 号 就 是 登录 数据 库 时 使 用 的 用 户 名 和 密码 ， 这 就 和 登录 操作 系统 一 样 ， 都 
需要 使 用 账号 和 密码 。 在 本 节 中 主要 讲解 如 何 使 用 SQL 语句 和 企业 管理 器 来 创建 和 管理 登 
录 账 号 。 


14.2.1 创建 登录 账号 


创建 登录 账号 首先 要 注意 的 是 账号 不 能 重 名 , 使 用 SQL 语句 创建 登录 账号 的 语法 形式 
如 下 : 

CREATE LOGIN loginname WITH PRSSWORD='password' 

[DEFAULT DATABASE=dbname] 

其 中 : 

口 loginname: 登录 名 。 

口 password: 密码 。 密 码 要 尽量 设置 复杂 一 些 。 

口 dbname: 指定 账户 登录 的 默认 数据 库 名 。 如 果 不 指定 默认 数据 库 名 ， 则 会 默认 将 

master 数据 库 作 为 默认 的 数据 库 。 

下 面 就 利用 上 面 学 习 的 创建 登录 账号 的 语法 来 创建 登录 账号 。 

【示例 1】 创建 登录 名 为 userl、 密 码 为 1a2b3c 的 登录 账号 。 

根据 题目 要 求 ， 创 建 登录 账号 的 语句 如 下 : 


CREATE LOGIN userl WITH PASSWORD='1la2b3c'; 


执行 上 面 的 语句 ， 效 果 如 图 14.1 所 示 。 

下 面 就 来 验证 userl 账号 是 否 能 成 功 登 录 。 在 登录 企业 管理 器 的 界面 中 ， 尝 试 输入 用 
户 名 userl 以 及 密码 1a2b3c， 效 果 如 图 14.2 所 示 。 

在 图 14.2 所 示 界 面 中 ， 单 击 “ 连 接 ” 按 钮 ， 即 可 登录 到 SQL Server 企业 管理 器 的 界面 
中 。 如 果 能 够 成 功 登 录 ， 那 就 说 明 创建 登录 账号 userl 成 功 了 。 
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站 
文件 虽 ” 编 辑 吕 ”视图 (查询 O) 项 目 B) 调试 D) 工具 ID 
窗口 (W) 社区 (QO 才 助 (H) 

卫 . 亲 于 查 放 N | 六 | 态 访 怨 | 记 | 芒 回 马 剖 忆 

a 4 | master -| 执行 WW 》 古渡 虹 
SQLQuery1.sql -.istrator (52))* 

| CREATE LOGIN userl WITH PASSWORD='1a2b3c'; 


己 消 息 | 

命令 已 成 功 完成 | 
| 2 
VER2006 (.,. | WIDW-9527\Administrato,..| master | 00:00:00 | 0 行 
行 S 列 ! chl 


图 14.1 创建 登录 账号 userl 
加 | 
Br 
< SQLServer2008 
服务 器 类 型 了); | 县 才 库 引 军 | 
服务 器 名称 G) iow-9527 WSSQLSERVER2008 了 
身份 验证 A); SQL Server 身份 验证 | 
熙 录 名 (中); se | 
密码 中): Fee 
厂 记 住 密码 如 


了 少 | 和 且 选项 @) 六 


图 14.2 使 用 userl 登录 界面 
全 说 明 : 除了 使 用 CREATE LOGIN 语句 可 以 创建 登录 账号 外 ， 还 可 以 通过 系统 存储 过 程 
sp_addlogin 来 创建 。 语 法 形式 如 下 : 
sp_addlogin username,userpwd,default database; 
这 里 ，username 是 登录 名 ，userpwd 是 密码 ，default database 是 为 用 户 指定 的 默 
认 数 据 库 。 如 果 用 sp_addlogin 创建 示例 1 的 用 户 账号 user1， 有 具体 语句 如 下 : 


sp_addlogin 'userl','la2b3c'; 


通过 上 面 的 语句 就 可 以 达到 与 示例 1 相同 的 效果 了 。 


14.2.2 ”修改 登录 账号 


创建 好 的 登录 账号 也 是 可 以 修改 的 ， 修 改 登录 账号 使 用 的 是 ALTER LOGIN 语句 。 使 
该 语句 可 以 修改 账号 名 、 密 码 以 及 默认 数据 库 等 信息 。 具 体 的 语法 形式 如 下 : 

ALTER LOGIN loginname 

[DISABLE | ENABLE] 

WITH 

{DEFAULT DATABASE =database|NAME=new_ login name|PASSWORD='password'} 


其 中 : 
口 loginname: 要 修改 的 账户 名 称 。 


“i 
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口 DISABLEIENABLE: 禁用 或 启用 账户 。 

口 database: 默认 数据 库 名 。 

口 new_login_name: 修改 后 的 账户 名 称 。 

口 password: 密码 。 

下 面 就 使 用 上 面 的 修改 登录 账号 的 语法 来 完成 如 下 的 示例 。 

【示例 2】 将 示例 1 中 的 登录 账号 userl 的 账号 名 修改 成 user2。 

根据 题目 要 求 ， 修 改 登 录 账 号 的 名 字 ， 具 体 的 语句 如 下 : 

ALTER LOGIN userl 

WITH 

NAME=user2; 

执行 上 面 的 语句 ， 效 果 如 图 14.3 所 示 。 

通过 图 14.3 所 示 的 效果 ， 就 完成 了 将 userl 改 成 user2 的 操作 。 不 用 忘记 了 ， 下 次 只 
能 使 用 user2 登录 了 。 

【示例 3】 将 登录 账号 user2 的 默认 数据 库 更 改 成 chapter14。 

首先 ， 要 在 SQL Server 中 新 建 数 据 库 chapter14。 然 后 再 修改 。 具 体 语 句 如 下 : 

CREATE DATABASE chapterl4; 

ALTER LOGIN user2 


WITH 
DEFAULT DATABASE = chapterl4; 


执行 上 面 的 语句 ， 效 果 如 图 14.4 所 示 。 

通过 图 14.4 所 示 的 效果 ， 可 以 看 出 将 user2 的 数据 库 设置 成 chapter14 了 。 

【示例 4】 将 登录 账号 user2 禁用 。 

禁用 账号 只 需要 在 ALTER 语句 中 使 用 DISABLE 关键 字 即 可 。 具 体 语句 如 下 : 

ALTER LOGIN user2 DISABLE; 

执行 上 面 的 语句 ， 效 果 如 图 14.5 所 示 。 

通过 图 14.5 所 示 的 效果 ， 就 将 user2 登录 名 禁用 了 。 如 果 想 启用 user2， 则 将 上 面 语句 
中 的 DISABLE 换 成 ENABLE 即 可 。 


本 Microsoft SQL SS 


ga 
文件 (E) 编 缉 (E) ”视图 W 查询 (9) 
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日 CREATE DATABASE chapteri4d, 
BE ALTER LOGIN user2 
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上 DEFAULT_DATABASE = 
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图 14.3 ”修改 登录 名 


就 绪 行 7 列 ! hi 


图 14.4 ”修改 默认 数据 库 


就 缩 
图 14.5 禁用 user2 
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外 说 明 : 不 仅 可 以 使 用 ALTER 语句 来 修改 登录 账号 ， 也 可 以 使 用 系统 存储 过 程 来 完成 。 
在 修改 登录 账号 密码 时 ， 可 以 使 用 系统 存储 过 程 sp_password 完成 ; 在 修改 登录 
账号 的 默认 数据 库 时 ， 可 以 使 用 系统 存储 过 程 sp_defaultdb。 


14.2.3 删除 登录 账号 


删除 登录 账号 就 很 简单 了 ， 使 用 DROP 语句 即 可 。 在 删除 账号 时 ， 只 要 知道 账号 名 就 
可 以 删除 。 但 是 ， 删 除 登 录 账 号 功能 是 不 可 逆 的 ， 因 此 ， 在 删除 前 一 定 要 确认 好 。 有 具体 的 
删除 语句 如 下 : 


DROP LOGIN login name; 


这 里 ，login_name 是 登录 账号 名 。 另 外 ,还 需要 注意 的 是 使 用 DROP LOGIN 语句 一 次 
只 能 删除 一 个 登录 账号 名 。 
下 面 就 使 用 示例 5 来 演示 如 何 删除 登录 账号 。 
【示例 S】 删除 登录 账号 user2 。 
只 需要 使 用 DROPLOGIN 即 可 删除 ， 有 具体 的 语句 如 下 : 


DROP LOGIN user2; 


执行 上 面 的 语句 ， 效 果 如 图 14.6 所 示 。 


通过 图 14.6 所 示 的 效果 , 看 到 账号 user2 已 经 成 功 删 。 ED 
除了 。 调式 (0D) 工具 (D 窗口 (WwW) 社区 (C) 帮助 (H) 


屋 加 VY | 记 | 过 记 四 | 户 | 切 名 
全 说 明 : 删除 登录 账号 也 可 以 使 用 系统 存储 过 程 。 | 
sp_droplogin 来 完成 。 章 pROP LOGIN userz 


14.2.4 ”使 用 企业 管理 器 管理 登录 账号 


读者 在 前 面 已 经 学 习 了 如 何 使 用 SQL 来 创建 和 管 
理 登 录 账号 , 现在 再 教 你 一 个 简单 的 方法 来 管理 登录 账 ”图 146 删除 登录 账号 user2 
号 ， 这 就 是 使 用 我 们 的 好 朋友 SQL Server 企业 管理 器 来 完成 。 在 企业 管理 器 中 ， 可 以 
很 容易 地 完成 登录 账号 的 创建 和 管理 的 操作 。 下 面 就 一 起 来 验证 使 用 企业 管理 器 的 便 
利 吧 。 


1. 创建 登录 账号 


-9527VAdministrato..，| master | 00:00:00 | 0 行 


就 绪 行列 1chl 才 


以 创建 登录 账号 userl 为 例 ， 在 企业 管理 器 中 创建 登录 账号 分 为 如 下 两 个 步 又。 

(1) 找到 创建 登录 账号 的 位 置 

在 “对 象 资源 管理 器 ”中 ， 展 开 “ 安 全 性 ”节点 ， 右 击 “ 登 录 名 ”选项 ， 在 弹出 的 右 
键 菜单 中 选择 “新 建 登录 名 ”选项 。 出 现 图 14.7 所 示 界 面 。 

(2) 填 入 登录 名 信息 
在 图 14.7 所 示 界 面 中 ， 填 入 登录 名 user1， 选 择 “SQL Server 身份 验证 ”方式 ， 并 设 
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由 登录 名 - 新 建 


服务 器 
ITDW-952TWWISSQLSERYER2008 


连接 
人 TDW-952TVAdninistrator 


圭 查看 连接 属性 


就 绪 删除 咏 
了 认 妆 据 库 D) 到 国 


图 14.7 新 建 登录 名 界面 


置 密码 。 在 默认 数据 库 处 选择 chapter14。 填 入 后 的 效果 如 图 14.8 所 示 。 
在 图 14.8 所 示 界 面 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 完成 userl 登录 账号 的 创建 。 


2. 修改 登录 账号 


以 更 改 登录 账号 userl 的 密码 为 例 ， 在 企业 管理 器 中 修改 登录 账号 及 密码 需要 如 下 两 
个 步骤 。 

(1) 找到 要 修改 的 登录 账号 

在 “对 象 资源 管理 器 ”中 ， 依 次 展开 “安全 性 ”|“ 登 录 名 ”节点 ， 右 击 userl 节点 ， 
在 弹出 的 右键 菜单 中 选择 “属性 ”选项 。 出 现 图 14.9 所 示 界 面 。 

(2) 修改 登录 账号 的 密码 

在 图 14.9 所 示 界 面 中 ， 修 改 密码 并 确认 密码 。 修 改 密码 后 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 
完成 登录 名 userl 的 修改 密码 操作 。 

读者 在 图 14.9 所 示 的 界面 中 会 发 现 只 能 修改 密码 、 默 认 数据 库 以 及 默认 语言 等 信息 ， 
但 是 无 法 修改 登录 名 。 那 么 ， 登 录 名 如 何 修改 呢 ? 很 简单 ， 如 果 要 修改 登录 名 可 以 直接 右 
击 要 修改 的 登录 名 ， 在 弹出 的 右键 菜单 中 选择 “ 重 命名 ”选项 ， 重 新 键入 登录 名 即 可 完成 
修改 操作 。 
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3. 删除 登录 账号 

以 删除 登录 账号 userl 为 例 ， 在 企业 管理 器 中 删除 登录 账号 需要 如 下 两 个 步骤 。 

(1) 找到 要 删除 的 登录 账号 

在 “对 象 资源 管理 器 ”中 ， 依 次 展开 “安全 性 ”|“ 登 录 名 ”节点 ， 右 击 userl 节点 ， 
在 弹出 的 右键 菜单 中 选择 “删除 ”选项 。 出 现 图 14.10 所 示 界 面 。 


= 将 | 


EEE 3 所 有 者 | 状态 ”| 


服务 器 
WIDN~9527\MSSQLSERVER2008 


连接 
MDW-952TVAdninistrator 


对 查看 连接 必 性 


图 14.10 ”删除 登录 账号 界面 


(2) 确认 删除 
在 图 14.10 所 示 界 面 中 ， 单 击 “确定 ”按钮 ， 即 可 完成 登录 账户 userl 的 删除 操作 。 


14.3 用 户 管 理 


读者 通过 14.1 节 的 学 习 , 应 该 体会 到 了 用 户 在 数据 库 安 全 性 中 的 重要 性 。 权 限 和 角色 
都 是 要 给 用 户 设置 的 。 用 户 管理 既 可 以 通过 SQL 语句 完成 , 也 可 以 通过 企业 管理 器 的 图 形 
界面 管理 。 在 本 节 中 ， 将 详细 地 讲解 如 何 管理 数据 库 的 用 户 。 


14.3.1 创建 用 户 


创建 用 户 是 用 户 管理 的 第 一 步 ， 数 据 库 中 的 用 户 也 是 不 能 够 重 名 ， 并 且 用 户 名 也 不 能 
够 以 数字 为 前 级 。 创 建 用 户 的 语法 形式 如 下 : 


CREATE USER user name [ { { FOR | FROM } 
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有 
LOGIN login _ name 


} 
| WITHOUT LOGIN 
] 


其 中 : 

口 user name: 用 户 名 。 指 定 登录 数据 库 的 用 户 名 。 

口 login_name: 指定 要 创建 数据 库 用 户 的 SQL Server 登录 名 。 

口 WITHOUT LOGIN: 指定 不 应 将 用 户 映射 到 现 有 登录 名 。 

这 里 ， 需 要 注意 的 是 如 果 在 创建 用 户 时 没有 指定 登录 名 ， 那 么 就 要 将 用 户 名 创建 成 与 
登录 名 同名 才 可 以 ， 否 则 会 出 现 错误 。 

【示例 6】 创建 用 户 testuser。 

为 了 创建 用 户 方便 , 首先 将 用 户 testuser 创建 成 登录 用 户 , 然后 再 使 用 CREATE USER 
语句 创建 用 户 。 具 体 的 语句 如 下 : 


CREATE LOGIN testuser WITH PASSWORD='123456'; 
CREATE USER testuser; 


执行 上 面 的 语句 ， 效 果 如 图 14.11 所 示 。 
通过 图 14.11 所 示 的 效果 ， 就 完成 了 用 户 testuser 的 创建 。 
【示例 7】 创建 在 数据 库 chapter14 上 的 用 户 testuserl 。 
在 本 例 中 登录 名 仍 使 用 前 面 创建 的 testuser， 创 建 用 户 testuserl 的 语句 如 下 : 
USE chapter14; =-- 指 定 要 创建 用 户 的 数据 库 
CREATE USER testuserl FOR LOGIN testuser; 
执行 上 面 的 语句 ， 效 果 如 图 14.12 所 示 。 
了 Microsoft SQL Server Management Studio =|Dlxl 


文件 (E) 编辑 (E) 视图 ( 几 。 查 询 (Q) 项目) 调试 D) 工具 D 
窗口 (W) ”社区 (帮助 (H) 


Rs Microsoft SQL Server Management Studio =Iolx| 
文件 四 六 纺 ( 视 国 WW) 查 六 0) 项 目 (E) 调试 D) 工具 DD 


窗口 W) 社区 (QO 帮助 中 


六 间 光 局 | 六 六 是 | 齐 | 芒 回 时 | 加: 
;一 六 | master >| 1 执 50W 》 时 部 站 


SQLQuery4.sql -istrat 
日 CREATE LOGIN 
| CREATE USER 


了 新 建 查询 六 六 沁 久 局 蕊 回归 加 瑟 
By 好 | hapterls “| 1 执 和 5 Pv 闻名 


SQLQuery4.sql wistrator (53))* vx 
日 us chapter14; ”-- 指 定 要 创建 用 户 的 
CREATE USER testuserl FOR LOGIN testuser; 


WER2008 (.,. | WIDW-9527\administrato, ,，| chapter14 | 00:00:00 | 0 行 
行 4 列 1 Chi 


图 14.11 创建 用 户 testuser 图 14.12 创建 用 户 testuserl 
通过 图 14.12 所 示 的 效果 ， 用 户 testuserl 就 创建 成 功 了 。 
全 说 明 : 创建 用 户 也 可 以 使 用 系统 存储 过 程 SP_GRANTDBACCESS 来 完成 。 


14.3.2 ”修改 用 户 


创建 好 用 户 后 ， 如 果 需 要 修改 用 户 的 信息 ,可 以 通过 ALTER USER 语句 来 修改 。 具体 
的 语法 形式 如 下 : 
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ALTER USER user name WITH 
| 
NAME=new username|LOGIN=loginname 


} 


其 中 : 

口 user name: 用 户 名 。 指 定 要 修改 的 用 户 名 。 

口 new_username: 修改 后 的 用 户 名 。 

口 loginname: 修改 后 的 登录 名 。 

下 面 就 使 用 示例 8 来 演练 如 何 修改 用 户 。 

【示例 8】 将 用 户 名 testuser 修改 成 newtestuser。 

根据 题目 要 求 ， 修 改 语句 如 下 : 

USE master; -打开 用 户 所 在 的 数据 库 

ALTER USER testuser WITH 

NAME=newtestuser; 

执行 上 面 的 语句 ， 效 果 如 图 14.13 所 示 。 

通过 图 14.13 所 示 的 效果 ， 用 户 testuser 就 修改 成 功 了 。 
14.3.3 ”删除 用 户 

删除 用 户 只 需要 用 户 名 就 可 以 搞定 了 ， 如 果 不 清楚 要 删除 的 用 户 名 ， 可 以 通过 系统 存 
储 过 程 SP_HELPUSER 来 查看 。 有 了 用 户 名 ， 使 用 下 面 的 删除 语句 即 可 完成 操作 。 

DROP USER username; 

这 里 ，username 就 是 要 删除 的 用 户 名 。 需 要 注意 的 是 在 删除 用 户 名 之 前 ， 先 要 将 用 户 
所 在 的 数据 库 使 用 USE 语句 打开 。 

下 面 就 使 用 示例 9 来 演练 如 何 删除 用 户 名 。 

【示例 9】 删除 数据 库 chapter14 中 的 用 户 名 testuserl 。 

根据 题目 要 求 ， 删 除 语句 如 下 : 


USE chapter147 
DROP USER testuserl; 


执行 上 面 的 语句 ， 效 果 如 图 14.14 所 示 。 


本 Microsof ft SQL Server Managenme 
文件 电 ”编辑 (E) ”视图 (VW) 查 济 (9) 项 目 (EB) 


=|D|x| 
文件 ”编辑 (E) 视图 ( 轨 。 查询 (Q) 项目) 调试 D) 工具 (D 


窗口 (Ww) 社区 (C) 科 助 bD 1 


和 助 (H) 
也 新 建 查询 六 | 总 凶 区 属 ™、 | 一 和 
启 | 贸 也 台 | 也 | 区 加 与 | 怠 遇 | 卫 章 建 相册 | 这 也 也 | 记 | 巧遇 
遇 3 | master =| 执行 Wj 》 更 w ?3 dl ~ oe 
SQLQuery4.sql -istrator (53))* vx 
lB 43: 
1 
dl » 
已 消息 
命令 已 成 功 完成 天 
a 2 
|sQLSERVER2008(.., | WIDW-9527\Administrato...| master | 00:00:00 | 0 行 -9527\VAdministrato...| chapter14 | 00:00:00 | 0 行 


就 绪 石 4 。 列 1 Ohi 


就 绪 行列 ch 


图 14.13 ”修改 用 户 testuser 图 14.14 ”删除 用 户 名 testuserl 
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从 图 14.14 所 示 的 效果 可 以 看 出 ， 用 户 名 testuserl 删除 成 功 了 。 


全 说 明 : 删除 用 户 也 可 以 通过 系统 存储 过 程 SP_REVOKEDACCESS 来 完成 。 


14.3.4 ”使 用 企业 管理 器 管理 用 户 


前 面 已 经 学 习 了 如 何 使 用 SQL 语句 来 创建 和 管理 用 户 , 实际 上 , 在 企业 管理 器 中 也 是 
可 以 管理 用 户 的 。 相 比 之 下 ， 使 用 企业 管理 器 会 变 得 更 容易 一 些 。 下 面 就 来 学 习 如 何 使 用 
企业 管理 器 管理 用 户 。 


1. 创建 用 户 


在 企业 管理 器 中 创建 用 户 很 方便 ， 通 过 以 下 3 个 步骤 即 可 完成 了 。 这 里 ， 以 创建 用 户 
userl 为 例 。 

(1) 打开 创建 用 户 的 界面 

在 “对 象 资源 管理 器 ”中 ,依次 展开 “数据 库 ”|chapater14|“ 安 全 性 ”节点 ， 右 击 “ 用 
户 ” 选 项 ， 在 弹出 右键 菜单 中 选择 “新 建 用 户 ” 选 项 ， 如 图 14.15 所 示 。 


=|9lxl 
i SN -Dew 
Ey 
不 安全 对 人 
子 扩展 习性 人 可 
人 5 要 录 名 中 [ 及 
个 下 书 各 陈 位 ) 
人 过 名 名 次 名 7 


(se 


就 二 drydat wrender 一 
三 aaeaydaterriter 
口 aa 恒 


[ele wa 
图 14.15 ”新 建 数 据 库 用 户 界面 
(2) 选择 登录 名 
在 图 14.15 所 示 界 面 中 ， 为 新 建 的 用 户 选择 一 个 登录 名 。 单 击 登 录 名 后 面 的 二 按钮， 
弹出 选择 登录 名 界面 ， 如 图 14.16 所 示 。 
在 图 14.16 所 示 界 面 中 ， 单 击 “ 浏 览 ” 按 钮 ， 出 现 图 14.17 所 示 界 面 。 
在 图 14.17 所 示 界 面 中 ， 选 中 一 个 登录 名 ， 单 击 “确定 ”按钮 ， 即 可 完成 登录 名 的 选 
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徊 选择 登录 各 x| 


选择 这 些 对 象 类 型 GE) ; 


图 14.16 ”选择 登录 名 


钥 查 找 对 萌 x 
找到 8 个 四 也 所 选 类 型 的 对 象 。 
匹配 的 对 象 电 ) 
|_ | 次 ES 
厂 欧 [tS_PolicyEventProcessineLoging#] 车 录 名 
FF [pS_PolicyTsqlExecutionLogin##] 登录 名 
厂区 Dr AMDRTTYVNETYORK SERVICE] 登录 名 


三 榴 Dr MTHORITI\SISTEN] 


图 14.17 浏览 用 户 界 面 


择 。 这 里 ， 选 择 登 录 名 testuser。 

(3) 填 入 用 户 信息 

完成 登录 名 的 选择 后 ， 填 入 用 户 名 user1， 还 可 以 为 用 户 选择 架构 或 者 角色 。 效 果 如 
图 14.18 所 示 。 

填 完 用 户 信息 后 ， 在 图 14.18 所 示 界 面 中 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 用 户 信息 的 


创建 。 
这 里 ， 关 于 角色 的 一 些 信 息 将 在 下 一 节 中 详细 讲解 。 
2. 修改 用 户 


有 了 创建 用 户 信息 的 操作 基础 ， 修 改 用 户 就 变 得 容易 多 了 。 在 企业 管理 器 中 ， 修 改 用 
户 信息 需 要 如 下 两 个 步骤 就 可 以 完成 了 。 在 企业 管理 器 中 修改 用 户 信 息 不 能 够 修改 用 户 的 
登录 名 。 

(1) 打开 用 户 信息 的 属性 界面 

在 “对 象 资源 管理 器 ”中 ， 依 次 展开 “数据 库 ”|chapater14|“ 安 全 性 ” |“ 用户” 节点 ， 
右 击 userl 用 户 ， 在 弹出 右键 菜单 中 选择 “属性 ”选项 ， 如 图 14.19 所 示 。 

(2) 修改 需要 的 信息 

在 图 14.19 所 示 界 面 中 ， 只 能 修改 userl 用 户 的 架构 以 及 角色 ， 修 改 后 单 击 “ 确 定 ” 按 
钮 ， 即 可 完成 对 用 户 信息 的 修改 操作 。 
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前 数 括 库 用 户 - 新 建 


图 14.18 ” 填 入 用 户 信息 


厂 ab_aatareader 
三 db_aaterriter 


图 14.19 用 户 userl 的 属性 界面 


如 果 需 要 修改 用 户 名 ， 可 以 直接 右 击 要 修改 的 用 户 名 ， 在 弹出 的 右键 菜单 中 选择 “ 重 
命名 ”选项 ， 重 新 键入 用 户 名 即 可 完成 修改 操作 。 
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3. 删除 用 户 


以 删除 用 户 名 userl 为 例 ， 在 企业 管理 器 中 删除 用 户 名 需要 如 下 两 个 步骤 。 

(1) 找到 要 删除 的 登录 账号 

在 “对 象 资源 管理 器 ”中 ， 依 次 展开 “数据 库 ”|chapater14|“ 安 全 性 ”| “用户 ” 节 点 ， 
右 击 userl 用 户 ， 在 弹出 的 右键 菜单 中 选择 “删除 ”选项 ， 如 图 14.20 所 示 。 


=|glx 


服务 器 
MIDI-352TVISSQLSERVER2008 
连接 
MID-9527\Adninistrator 


惠 查看 连接 尾 性 


图 14.20 ”删除 用 户 界 面 


(2) 确认 删除 
在 图 14.20 所 示 界 面 中 ， 单 击 “确定 ”按钮 ， 即 可 完成 用 户 userl 的 删除 操作 。 


14.4 角色 管理 


在 使 用 企业 管理 器 创建 用 户 时 ， 在 创建 用 户 的 界面 上 看 到 了 数据 库 角色 的 设置 。 数 据 
库 角色 是 系统 自 带 的 ， 通 过 这 些 角色 就 可 以 给 用 户 赋予 一 些 权 限 。 在 本 节 中 就 将 带领 读者 
认识 数据 库 角色 所 包含 的 权限 以 及 如 何 自 定义 角色 。 


14.4.1 认识 常用 角色 


在 安装 SQL Server 数据 库 后 ， 系 统 就 会 为 数据 库 配 备 一 些 常用 的 角色 。 为 用 户 设置 角 
色 就 相当 于 是 将 一 组 权限 一 起 设置 给 用 户 了 ， 既 方便 了 操作 又 能 避免 再 赋予 权限 时 出 现 错 
误 。 下 面 就 来 一 起 看 看 数据 库 中 常见 的 角色 吧 ， 如 表 14-1 所 示 。 
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表 14-1 数据 库 角色 


数据 库 角 色 说 明 
db accessadmin 拥有 添加 或 删除 用 户 的 权限 
db securityadmin 拥有 管理 全 部 权限 、 对 象 所 有 权 和 角色 的 权限 


db ddladmin 
db_backupoperator 
db datareader 


拥有 DDL 操作 权限 
拥有 执行 DBCC、CHECKPOINT 和 BACKUP 语句 的 权限 
拥有 选择 数据 库 内 任何 用 户 表 中 的 所 有 数据 的 权限 


db datawriter 


拥有 更 改 数据 库 内 任何 用 户 表 中 的 所 有 数据 的 权限 


db _ owner 


拥有 全 部 权限 


db denydatareader 


禁止 选择 数据 库 内 任何 用 户 表 中 的 任何 数据 


db denydatawriter 


禁止 更 改 数据 库 内 任何 用 户 表 中 的 任何 数据 


读者 看 过 这 张 表 的 解释 ， 是 否 对 这 些 角色 有 种 似曾相识 的 感觉 。 没 错 ， 在 图 14.19 所 


示 界 面 中 的 人 


所 部 分 就 已 经 遇 到 过 了 。 通 过 表格 中 对 这 些 角色 的 解释 ， 读 者 就 可 以 在 创建 


用 户 时 有 的 放 矢 地 设置 了 。 如 果 在 创建 用 户 时 ， 没 有 指定 角色 ， 默 认 都 是 public 类 型 的 角 


色 。public 角 


色 是 在 每 个 数据 库 中 都 存在 的 ， 并 且 该 角色 是 不 能 删除 的 。 


外 说 明 : 如 果 要 查看 数据 库 的 固定 角色 ,可 以 通过 系统 存储 过 程 SP_HELPFIXEDROLE 来 
查看 。 


14.4.2 ”创建 角色 


在 SQL Server 数据 库 中 ， 如 果 表 14-1 中 所 列 出 的 数据 库 角色 还 不 能 够 满足 您 的 要 求 ， 
还 可 以 自 定义 角色 。 自 定义 角色 使 用 CREATE ROLE 语句 完成 。 具 体 的 语法 形式 如 下 : 


CREATE ROLE role name [AUTHORIZATION owner name]; 


其 中 ， 


口 role name: 角色 名 称 。 该 角色 名 称 不 能 与 数据 库 固定 角色 的 名 称 相同 。 
口 owner_name: 用 户 名 称 。 角 色 所 作用 的 用 户 名 称 。 如 果 省 略 了 该 名 称 ， 角 色 就 被 
创建 到 当前 数据 库 的 用 户 上 。 

通过 上 面 的 语句 所 创建 的 角色 并 没有 设置 权限 ， 只 是 创建 了 一 个 角色 名 称 而 已 。 不 用 
着 急 ， 在 14.5 节 中 就 会 学 习 如 何 给 角色 赋予 权限 了 。 

【示例 10】 创建 作用 在 数据 库 chapter14 上 userl 用 户 的 角色 rolel 。 

根据 题目 要 求 ， 创 建 语句 如 下 : 

USE chapterl4d; 

CREATE ROLE rolel AUTHORIZATION userl; 

执行 上 面 的 语句 ， 效 果 如 图 14.21 所 示 。 

通过 图 14.21 所 示 的 效果 , 就 可 以 看 出 角色 rolel 创建 成 功 。 如果 要 查看 数据 库 是 否 存 


在 这 个 角色 ， 


可 以 通过 系统 存储 过 程 SP_HELPROLE 来 查看 。 查 询 效 果 如 图 14.22 所 示 。 


从 图 14.22 所 示 的 结果 可 以 看 出 ， 角 色 rolel 位 于 第 2 行 。 在 使 用 SP_HELPROLE 查 
询 时 ， 如 果 当 前 数据 库 不 是 要 查询 的 数据 库 ， 就 要 先 使 用 USE 语句 打开 要 查询 的 数据 库 。 
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[vcrosoft SQL Server Management si le E 
文昌 ”编辑 日 ”视图 查询 (o) 项 目 (p) 
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SQLQuery1.sql -istrator (52))* vx 
日 spP_HELPROLE 


[Rs werosor Sal Server Management Sudo JE 
文件 编辑 (6 ”视图 W 查询 (Q) 项 目 B) 调 坛 D) 工具 0D 
窗口 (Ww) ”社区 (CO) 帮助 由 

;也 新建 查询 | 让 | 人 也 下 | 包 | 蕊 加 本 | 受训 
; MY i | chapter14 "| 执行 WW) 》 w 


SQLQuery1.sql -istrator (52))* 
日 05SE chapterl 
| CREATE ROLE rolel AUTHORIZATION userl; 


4 

耳 消 息 | 
命令 已 成 功 完成 。 

YER2008 (,,，| WIDW-9527\Administrato,.，| chapter14 | 00:00:00 | 0 行 

就 绪 行 5 “ 列 ! Chl 


db_ddadmin 

db_backupoperator 16389 
16390 
16391 
16392 
16393 


MIDw-9527\administrato,, ,| chapter14 | 00:00:00 | 11 行 
就 绪 行 3 列 1 Chl /4 


图 14.21 创建 角色 rolel 图 14.22 ”查询 数据 库 chapter14 中 的 角色 (示例 10) 


14.4.3 修改 角色 


修改 角色 也 是 很 容易 的 ， 但 是 只 能 修改 角色 的 名 称 ， 其 他 的 内 容 是 不 能 够 修改 的 。 修 
改 角 色 使 用 的 语句 是 ALTER ROLE， 具 体 的 语法 形式 如 下 : 

ALTER ROLE role name 

WITH NAME=new_ name; 

其 中 : 

口 role_ name: 要 修改 的 角色 名 称 。 

口 new_name: 修改 后 的 角色 名 称 。 

【示例 11】 将 示例 10 中 创建 的 角色 rolel 更 名 成 role11。 

根据 题目 要 求 ， 修 改 语句 如 下 : 

ALTER ROLE rolel 

WITH N; =rolell; 

执行 上 面 的 语句 ， 效 果 如 图 14.23 所 示 。 

下 面 使 用 系统 存储 过 程 SP_HELPROLE 查看 数据 库 chapter14 中 的 角色 ， 看 看 是 否 更 
改 成 功 了 。 查 询 效 果 如 图 14.24 所 示 。 

从 图 14.24 所 示 的 效果 可 以 看 出 ， 角 色 rolel 已 经 更 改 成 了 rolel1 。 


14.4.4 删除 角色 


删除 角色 就 更 容易 了 , 只 要 知道 角色 名 称 就 可 以 删除 了 。 如 果 记 不 清 要 删除 哪个 角色 ， 
还 可 以 通过 之 前 用 过 的 系统 存储 过 程 SP_HELPROLE 来 查看 。 删 除 角 色 的 语法 形式 如 下 : 


DROP ROLE role name; 


“Me 
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项 目 (EB) 调试 Dj 工具 (D 窗口 (WwW) 


社区 (QO 帮助 (由 

巡 新 时 IY | 让 记过 王 | 记 | 蕊 副 

WR | dapterl4 ~ 吕 

SQLQuery1.sql .istrator (52))* vx 

站 sp_HELPROLE ~ 
6 可 | 


文件 (E) ”编辑 (E) ”视图 (查询 (9) 
项 目 (C) 调式 0) 工具 (D 窗口 (Ww) 
社区 (CO 必 助 (由) 

卫 .和 尘 查 芭 U | 让 | 也 也 本 | 记 中 
; MY a | chapter14 ~ 局 
SQLQuery1.sql .istrator (52))* 
日 ALTER ROLE rolel 
wz NANE=rolell; 


527Vadministrato, .| chapter14 | 00:00:00 | 11 行 
就 绪 


第 ! 第 12 4 
图 14.23 ”修改 角色 rolel 图 14.24 ”查询 数据 库 chapter14 中 的 角色 (示例 11) 
这 里 ，role_name 是 要 删除 的 角色 名 称 。 在 删除 角色 前 ， 还 要 使 用 USE 语句 打开 要 删 
除 角色 所 在 的 数据 库 。 


【示例 12】 删除 在 chapter14 中 创建 的 角色 rolell 。 
根据 题目 要 求 ， 删 除 语句 如 下 : 


USE chapterl4; 
DROP ROLE rolell; 


执行 上 面 的 语句 ， 效 果 如 图 14.25 所 示 。 
下 面 使 用 系统 存储 过 程 SP_HELPROLE 验证 是 否 将 rolell 删除 了 。 验 证 效果 如 图 14.26 
所 示 。 


[Emerosort Sor server Manonerieri= I PE] 
文件 (日 编辑 (ED 视 医 (W 查询 (Q) 项目) 
调式 (D0) 工具 (TD) 窗口 Ww) 社区 (CO) 帮助 (H) 


也 新建 查 六 | 车 六 证 | 访 | 蕊 中 
By 38 | chapter14 = 忆 


“Luery1sql ~istrator (52))* 
| 章 sp_HELPROLE 


着 
文件 人。 编外 (E) 视图 查询 (o) 
项 目 (C) 调 斌 D) 工具 (D 窗口 (w) 
社区 人 必 助 中 
六 新 当 吉 汐 Y | 户 | 售 六 加 | 已 台 
WY 8 | chapter14 zx a 
SiOuery sal strator Ca) | 
USE chapter14; 

DROP ROLE roleil; 


W9527\Administrato..,| chapter14 | 00:00:00 | 10 行 
就 绪 第 1 各 1 珊 才 


图 14.25 ”删除 角色 rolell 图 14.26 查询 数据 库 chapter14 中 的 角色 (示例 12) 
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从 图 14.26 中 的 效果 可 以 看 出 ， 角 色 rolel1 确实 被 删除 了 。 
14.4.5 ”使 用 企业 管理 器 管理 角色 
在 前 面 的 几 个 小 节 中 已 经 学 习 了 如 何 使 用 SQL 语句 来 创建 和 管理 角色 。 尽 管 使 用 SQL 
语句 来 管理 角色 已 经 很 容易 了 ， 但 是 还 是 很 有 必要 了 解 在 企业 管理 器 中 管理 角色 。 这 是 在 
一 时 想 不 起 创建 语法 时 的 一 个 解决 方案 哦 ! 下 面 就 来 学 习 如 何在 企业 管理 器 中 创建 、 修 改 
以 及 删除 角色 。 


1. 创建 角色 


在 企业 管理 器 中 ， 创 建 角 色 分 为 如 下 几 个 步 又。 这 里 以 创建 角色 rolel 为 例 。 
(1) 找到 新 建 角色 的 界面 
在 “对 象 资 源 管理 器 ”中 ， 依 次 展开 “数据 库 ”|chapter14|“ 安 全 性 ” |“ 角色 ”节点 ， 


右 击 “ 数 据 库 角色 ”选项 , 在 弹出 的 右键 菜单 中 选择 “新 建 数据 库 角 色 ” 选 项 ,弹出 图 14.27 
所 示 界 面 。 


服务 器 
TDY-952TVISSQLSERYEE2006 


连接 
TDW-952T\Adninistrator 
于 查看 连 按 属性 


就 请 


EI 


ww | w | 
图 14.27 新 建 数 据 库 角色 
(2) 填 入 角色 信息 
在 图 14.27 所 示 界 面 中 ， 可 以 为 角色 设置 名 称 、 所 有 者 以 及 其 他 角色 信息 。 这 上 
色 名 称 rolel1， 并 选择 所 有 者 user1， 填 入 效果 如 图 14.28 所 示 。 
填 入 角色 信息 后 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 角色 rolel 的 创建 。 


入 角 


[IE 


“0 
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图 14.28 填 入 角色 信息 


2. 修改 角色 


在 企业 管理 器 中 ， 修 改 角色 也 是 通过 两 个 步骤 就 可 以 完成 的 。 但 是 ， 在 企业 管理 器 的 
界面 中 是 不 能 修改 角色 名 称 的 ， 只 能 修改 角色 所 有 者 等 信息 。 如 果 想 修改 角色 名 称 ， 读 者 
还 得 使 用 SQL 来 修改 的 。 这 里 仍然 以 修改 rolel 为 例 。 具 体 的 步骤 如 下 所 示 。 

(1) 找到 修改 角色 的 界面 

在 “对 象 资源 管理 器 ”中 ， 依 次 展开 “数据 库 ”|chapter14|“ 安 全 性 ”| “角色 ”|“ 数 
据 库 角 色 ” 节 点 ， 右 击 rolel 选项 ， 在 弹出 的 右键 菜单 中 选择 “属性 ”选项 ， 弹 出 图 14.29 
所 示 界 面 。 

(2) 修改 角色 信息 

在 图 14.29 所 示 界 面 中 ,修改 rolel 的 相关 信息 。 然 后 单 击 “ 确 定 ” 按 钮 即 可 完成 角色 
的 修改 操作 。 


3. 删除 角色 


在 企业 管理 器 中 ， 删 除 角色 的 操作 是 角色 管理 中 最 简单 的 一 个 操作 了 ， 也 最 能 体现 企 
业 管 理 器 的 便利 性 。 下 面 仍然 以 删除 角色 rolel 来 演示 如 何 删除 角色 。 有 具体 的 步骤 如 下 
所 示 。 

在 “对 象 资源 管理 器 ”中 ， 依 次 展开 “数据 库 ”|chapter14|“ 安 全 性 ”|“ 和 角色 ”|“ 数 
据 库 角色 ”节点 ， 右 击 rolel 选项 ， 在 弹出 的 右键 菜单 中 选择 “删除 ”选项 ， 弹 出 图 14.30 
所 示 界 面 。 
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重 煞 闯 库 角色 屋 性 - rolel 


14.29 数据 库 角色 属性 


删除 对 象 


图 14.30 ”删除 角色 确认 界面 
在 图 14.30 中 ， 单 击 “ 确 定 ” 选 项 ， 即 可 将 角色 rolel 删除 了 。 


“305。 


第 4 篇 数据 库 的 管理 


14.5 权限 管理 


终于 到 了 本 章 的 核心 问题 了 ， 前 面 讲解 过 的 用 户 、 角 色 都 是 要 通过 权限 的 设置 来 确保 
数据 库 安全 的 。 本 节 中 就 将 讲解 如 何 给 用 户 或 角色 设置 权限 。 


14.5.1 授予 权限 


GRANT 语句 用 来 对 主体 授予 安全 对 象 的 权限 ， 该 权限 包括 是 否 允 许 访问 当前 数据 库 
的 表 、 视 图 等 对 象 。GRANT 的 常用 的 语法 如 下 : 


GRANT Permission [ON table_namelview_name] TO user name|role name 
WITH GRANT OPTION 


其 中 : 

口 permission: 权限 名 称 。 

口 table namelview_name: 表 名 或 视图 名 。 

口 user_namelrole name: 用 户 名 或 角色 名 。 

口 WITH GRANT OPTION: 表示 权限 授予 者 可 以 向 其 他 用 户 授 予 权限 。 

下 面 就 使 用 授予 权限 的 语法 来 演练 缕 ! 请 看 示例 13。 

【示例 13】 给 用 户 userl 在 数据 库 chapter14 中 授予 创建 表 的 权限 。 

根据 题目 要 求 ， 授 予 权限 的 语句 如 下 : 

USE chapterl4; 

GRANT create table TO userl; 

执行 上 面 的 语句 ， 效 果 如 图 14.31 所 示 。 

通过 图 14.31 所 示 的 效果 ， 就 成 功 地 为 用 户 userl 授予 了 创建 表 的 权限 。 查 询 用 户 拥 
有 的 权限 可 以 通过 系统 存储 过 程 sp_helprotect 来 完成 。 查 询 userl 的 权限 效果 如 图 14.32 
所 示 。 


了、Microsoft SQL Server Management Studia -ID|x| 


=I9lx| en OW 
文件 编辑 E) 视图 查询 Ko) 项目) 调 荆 D) 9 网 

工具 D 窗口 (社区 (QO 才 助 中 也 间 寻 查询 | 让 | 也 应 加 | 证 | 区 加 己 | 副 下 

也 .新建 查 询 几 | 让 | 也 也 瑟 | 记 | 记 加 本 到 ER | 中 hapterl4 "| 执行 WwW mv ?3 邓 [ 回 | 8” 总 
MY Wa | chapter14 =| 执行 W) 》 半 5QLQuery1.sql -istrator (53))* | vx 
er sp_helprotect Busername='userl'; 习 


USE chapter14; 
GRANT create table TO userl; 


后 消息 | 
命令 已 成 功 完 成 。 : 
dl 


Dw-9527\M55QLSERVER2008 (| WIDW-9527\Administrato,,，| chapter14 | 00:00:00 | 2 行 
就 绪 行 4 列 1 chl 


图 14.31 给 用 户 授予 权限 图 14.32 查询 userl 的 权限 
从 图 14.32 所 示 的 效果 ， 可 以 看 出 userl 已 经 被 授予 了 Create Table 的 权限 。 
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14.5.2 ”拒绝 权限 


成 ， 


所 谓 拒 绝 权限 ,就 是 指 让 数据 库 对 象 不 具备 某 种 权限 。 拒绝 权限 使 用 DENY 语句 来 完 
它 与 授予 权限 的 语法 形式 类 似 。 具 体 的 语法 形式 如 下 : 

DENY Permission [ON table name|lview name] TO user name|role name 
WITH GRANT OPTION 

其 中 : 

口 permission: 权限 名 称 。 

口 table namelview_name: 表 名 或 视图 名 。 

口 user_namelrole_name: 用 户 名 或 角色 名 。 

口 WITH GRANT OPTION: 表示 权限 授予 者 可 以 向 其 他 用 户 授 予 权限 。 

下 面 就 使 用 示例 14 来 演示 如 何 对 用 户 设 置 拒绝 权限 。 

【示例 14】 设置 用 户 userl 在 数据 库 chapter14 中 不 能 创建 视图 。 

根据 题目 要 求 ， 要 为 userl 设置 拒绝 权限 Create View， 具 体 的 语句 如 下 : 


USE chapterl4; 
DENY create view TO userl; 


执行 上 面 语句 ， 效 果 如 图 14.33 所 示 。 
通过 图 14.33 所 示 的 效果 ,就 已 经 使 用 户 userl 不 能 够 再 创建 视图 了 。 下 面 就 通过 系统 


存储 过 程 sp_helprotect 来 查看 用 户 userl 当前 所 具有 的 权限 。 效 果 如 图 14.34 所 示 。 


Microsoft SQL Server Management Studig 本 glxj 
文件 (上 编辑 (E) ”视图 (V) 查 涵 (Q) 项目) 调试 D) 工具 (D 窗口 (Ww) 
社区 (C) 帮助 td) 
EM Microsoft SQL Server Management std T= EE 也 新建 查询 六 | 售 渤 串 | 启 | 芒 回 时 | 况 忆 
文件 (E) ”编辑 (E) 视图 查询 (o) 项 目 () 世家 | chapterl4 "| ?执行 WW 》 a v 江 申 | 国 | 
调式 (0) 工具 ID 窗口 (Ww) 社区 (CO 帮助 St OueryL.sql istrator (53))*| 人 
及 新 洁 吉 测 VY | 让 | 轧 也 四 | 记 | 记 中 中 站 sp_helprorecc Busername='userl' 习 


RY Wy | chapter14 


SQLQuery1.sql -istrator (53))* 
USE chapter14 
DENY create View TO use 


Dery Create View 
Grant CONNECT 
Grant Create Table 


吕 

已 消息 | 

命令 已 成 功 完成 。 当 
IWIDW-9527\VAdrministrato,.，| chapter14 | 00:00:00 | 0 行 


B27\mssQLSERVER2008(.,,| WIDW-9527\Administrato,,，| chapter14 | 00:00:00 | 3 行 


就 绪 行 5 列 2 ch2 才 就 绪 行 4 列 1 chi >| 


图 14.33 ”给 用 户 设置 拒绝 权限 图 14.34 查看 用 户 userl 的 权限 
从 图 14.34 可 以 看 出 ，userl 新 增 了 一 个 Create View 的 拒绝 权限 。 


14.5.3 ”收回 权限 


收回 权限 ,读者 从 字面 上 的 意思 就 应 该 能 够 猜 出 来 是 将 原 有 的 权限 取消 ,在 SQL Server 


数据 库 中 ， 收 回 权 限 使 用 的 是 REVOKE 语句 。 它 既 能 够 取消 数据 库 对 象 的 授予 权限 ， 也 
能 够 取消 其 拒绝 权限 。 具 体 的 语法 形式 如 下 : 


REVOKE permission [ON table namelview name] TO user namel|role name 
WITH GRANT OPTION 
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其 中 : 

口 _ permission: 权限 名 称 。 

口 table namelview_name: 表 名 或 视图 名 。 

口 user namelrole name: 用 户 名 或 角色 名 。 

口 WITH GRANT OPTION: 表示 权限 授予 者 可 以 向 其 他 用 户 授予 权限 。 

下 面 就 使 用 示例 15 来 演示 如 何 收回 权限 。 

【示例 15】 将 用 户 userl 在 示例 13 中 被 授予 的 权限 Create Table 收回 。 

根据 题目 要 求 ， 具 体 的 语句 如 下 : 

USE chapPter147 

REVOKE create table TO userl; 

执行 上 面 的 语句 ， 效 果 如 图 14.35 所 示 。 

通过 图 14.35 所 示 的 效果 ， 可 以 看 出 userl 的 Create Table 权限 被 收回 了 。 下 面 就 通过 
系统 存储 过 程 sp_helprotect 来 检验 userl 用 户 是 否 存 在 Create Table 权限 。 效 果 如 图 14.36 
所 示 。 


有 、 Microsoft SQL Server Management SHUdid =Iojx|l 


Rs Microsoft SQL Server Management Studig IE| 
文件 编辑 ( 昌 视图 VW) 查询 Q) ”项目 (E) 调式) 工具 (D 窗口 (Ww) 


文件 人 编辑 (E) 视图 查询 (Q) ”项目 (P) 调试 DO) 

工具 (D ”窗口 (W) 社区 (帮助 (H) 社区 (QO) 帮助 (中 

也 新建 查询 疡 仙 也 四 记功 加 要 过 三 也. 章 建 查询 | 让 这 也 加 | 记 | 蕊 台 己 怠 扣 

My 3 | chepterl4 |! >》 B82 | chapter4 "| 执 和 0 有 v 叹 问 国 | 站 六 
5QLQuery1.sql .istrator (53))* -x SQLQuery1.sql .istrator (53))* | vx 


USE chapterl4; 
| REVOKE creare table TO userl; 


闫 sp_helprocecc Busername='userl'; 


Create View 


局 消息 | 
命令 已 成 功 完成。 习 
dl 2 


CONNECT 


» 
I08 (,,,| WIDW-9527\Administrato,,，| chapter14 | 00:00:00 | 0 行 
就 绪 行 5 列 ! Chl 


图 14.35 收回 用 户 userl 的 Create Table 权限 图 14.36 ”查询 用 户 userl 的 权限 


从 图 14.36 所 示 的 效果 可 以 看 出 , 用 户 userl 已 经 没有 创建 表 的 权限 了 。 读者 可 以 按照 
上 面 的 语句 将 用 户 userl 的 拒绝 权限 Create View 收回 ， 然 后 再 使 用 系统 存储 过 程 
sp_helprotect 验证 是 否 将 其 权限 收回 了 。 

在 SQL Server 数据 库 中 ， 权 限 是 不 能 够 独立 创建 的 ， 全 部 都 是 在 其 用 户 或 角色 上 进行 
操作 的 ， 因 此 ， 企 业 管 理 器 对 权限 的 管理 ， 在 创建 用 户 或 角色 的 时 候 就 已 经 被 使 用 了 。 这 
里 ， 就 不 再 使 用 企业 管理 器 对 权限 进行 重复 操作 了 。 


14.6 本 章 小 结 


在 本 章 中 主要 讲述 了 SQL Server 中 几 个 常用 的 数据 库 安 全 对 象 , 包括 : 登录 账户 、 用 
户 、 角 色 以 及 权限 。 在 登录 账户 部 分 主要 讲解 了 如 何 创建 、 修 改 以 及 删除 登录 账户 ; 在 用 
户 部 分 主要 讲解 了 创建 用 户 时 必须 要 使 用 登录 账户 以 及 如 何 修改 和 删除 用 户 ; 在 角色 部 分 
主要 讲解 了 角色 的 创建 、 修 改 以 及 删除 ， 在 权限 部 分 主要 讲解 了 给 用 户 或 角色 授予 、 拒 绝 
以 及 收回 权限 的 操作 。 读 者 在 学 习 完 本 章 内 容 后 ， 应 该 学 会 合理 使 用 用 户 或 角色 ， 并 通过 
给 予 适 合 的 权限 来 提高 数据 库 的 安全 性 。 
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14.7 本 章 习 题 
一 、 填 空 题 


1. 授予 权限 的 语句 是 
2. 创建 角色 的 语句 是 
3. 创建 登录 账号 的 语句 是 
二 、 选 择 题 
1. 下 面 的 关键 字 中 哪个 是 收回 权限 的 关键 字 
A. CREATE B. GRANT C. REVOKE D. 以 上 都 不 是 
2. 下 面 哪个 语句 是 用 来 创建 用 户 的 E 
A. CREATE USER B. CREATE USERS 
C. CREATETABLE D. 以 上 都 不 是 
3. 下 面 对 角 色 的 描述 中 正确 的 是 有 
A. 在 SQL Server 数据 库 中 ， 角 色 与 用 户 是 同一 个 意思 
B. 在 SQL Server 数据 库 中 ,角色 可 以 理解 成 权限 的 一 个 集合 ， 可 以 通过 角色 给 用 
户 授 予 权限 
C. 在 SQL Server 数据 库 中 ， 角 色 就 是 权限 
D. 以 上 都 不 对 


三 、 问 答题 

1. 登录 账户 与 用 户 有 什么 区 别 ? 
2. 角色 的 好 处 是 什么 ? 

3. 角色 和 权限 的 关系 是 什么 ? 
四 、 操 作 题 


1. 为 数据 库 chapter14 创建 名 为 loginl 的 登录 账户 。 

2， 使 用 登录 用 户 login1 为 数据 库 chapter14 创建 用 户 userl 。 
3. 为 用 户 userl 授予 修改 表 的 权限 。 

4. 收回 用 户 userl 修改 表 的 权限 。 


“Me 
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数据 库 的 备份 和 恢复 ， 是 SQL Server 数据 库 中 一 个 必 备 的 功能 。 有 了 数据 库 的 备份 和 
还 原 功能 , 就 能 够 更 灵活 地 保护 和 使 用 数据 。 当 需要 将 数据 库 换 到 另 一 台 计算 机 上 使 用 时 ， 
只 需要 将 原 有 的 数据 库 进 行 备份 ， 并 还 原 到 目标 计算 机 即 可 。 本 章 将 详细 讲述 数据 库 备 份 
和 还 原 的 具体 操作 方法 。 

本 章 的 主要 知识 点 如 下 : 

口 如 何 备份 数据 库 

口 如 何 还 原 数据 库 

口 如 何 分 离 和 附加 数据 库 


15.1 数据 库 备份 

数据 库 备 份 就 与 平时 我 们 在 计算 机 中 复制 文件 是 一 样 的 ,只 不 过 略 显 复杂 一 些 。 在 SQL 
Server 中 备份 数据 库 可 以 使 用 SQL 语句 也 可 以 使 用 企业 管理 器 。 下 面 就 为 读者 一 一 道 来 。 
15.1.1 数据 库 备份 的 类 型 

数据 库 备份 也 是 分 为 多 种 类 型 的 ， 只 有 掌握 了 备份 的 类 型 ， 才 能 够 合理 地 对 数据 库 进 
行 备份 。 在 SQL Server 2008 中 ， 数 据 库 备份 主要 分 为 如 下 4 种 类 型 。 

1. 完整 备份 

所 谓 完整 备份 ， 就 是 将 整个 数据 库 的 文件 全 部 备份 了 。 通 常数 据 库 是 由 数据 文件 和 日 
志文 件 组 成 的 ， 也 就 是 说 完整 备份 就 是 将 这 些 文件 进行 了 备份 。 通 过 对 数据 库 进 行 完整 备 
份 ， 可 以 将 数据 库 恢复 到 备份 时 的 状态 。 当 对 数据 库 进行 完整 备份 时 ， 是 对 数据 库 当 前 的 
状态 进行 备份 ， 不 包括 任何 没有 提交 的 事务 。 

2. 事务 日 志 备份 

事务 日 志 备份 主要 就 是 对 数据 库 中 的 日 志 进 行 的 备份 ， 也 就 是 记录 所 有 数据 库 的 变 
化 。 事 务 日 志 备 份 也 相当 于 一 次 完整 数据 库 备份 ， 通 过 事务 日 志 备 份 也 能 够 将 数据 库 恢 复 
到 备份 状态 ， 但 是 不 能 还 原 完 整 的 数据 库 。 

3. 差异 备份 

所 谓 差异 备份 , 就 是 指 备 份 数据 库 中 每 次 变化 的 部 分 。 差 异 备 份 能 够 提高 备份 的 效率 ， 
每 次 只 备份 一 部 分 修改 过 的 内 容 ， 而 不 必 全 部 备份 。 通 过 差异 备份 可 以 提高 备份 的 效率 ， 
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同时 减少 了 备份 所 占用 的 空间 。 
4. 文件 及 文件 组 备份 


前 面 已 经 学 习 过 ， 数 据 库 文件 是 存放 在 文件 或 文件 组 中 的 。 因 此 ， 备 份 数据 库 时 也 可 
以 通过 选择 文件 或 文件 组 ， 对 数据 库 进行 备份 。 通 过 指定 文件 或 文件 组 能 够 节省 备份 数据 
的 空间 。 当 然 ， 对 于 小 型 数据 库 ， 通 常 就 不 用 采用 文件 及 文件 组 的 备份 方式 了 。 文 件 及 文 
件 组 的 备份 方式 通常 用 于 数据 量 巨大 的 数据 库 的 。 


15.1.2 备份 数据 库 


上 一 小 节 已 经 讲解 了 备份 的 常用 类 型 ， 备 份 数据 库 经 常会 使 用 完整 备份 和 差异 备份 两 
种 方式 。 常 用 的 语法 形式 如 下 : 


BACKUP DATABASE database 

TO DISK='path' 

[WITH DIFFERENTIAL] 

其 中 

口 database: 要 备份 的 数据 库 名 。 

口 path: 数据 库 备 份 的 目标 文件 ， 数 据 库 备份 文件 的 扩展 名 是 .bak。 例 如 : 备份 到 
Ci\data\a.bak。 

口 WITH DIFFERENTIAL: 差异 备份 数据 库 。 省 略 该 语句 ， 则 执行 的 是 完整 备份 数 
据 库 。 

有 了 备份 数据 库 的 语法 规则 ， 就 可 以 对 数据 库 进行 完整 备份 和 差异 备份 了 。 下 面 就 分 

别 使 用 示例 1 和 示例 2 来 演示 如 何 对 数据 库 进 行 备份 。 

【示例 1】 创建 数据 库 chapter15， 并 对 其 进行 完整 备份 。 

根据 题目 要 求 ， 语 句 如 下 : 

CREATE DATABASE chapter15; =-- 创 建 数据 库 chapter15 


BACKUP DATABASE chapter15 
TO DISK='D: \backdata\chapter15 back.bak'; 


执行 上 面 的 语句 ， 效 果 如 图 15.1 所 示 。 


Microsoft SQL Server Management Studio 识 -IOlx|l 
文件 (编辑 (E) ” 视 国 (W 查询 (9) 项 目 (P) 调式 D) 工具 (D 窗 D(W 社区 (QO 


帮助 (H) 

BE EE 

BY Qa | master -| 执 和 5 下 v 品 剖 国 / 玉 弘 | 泣 图 名 
SQLQuery1.sql -..istrator (52))*| vx 


日 BhMCKRUP DATABASE chapter15 
| DISK='D:\backdata\ chapter15 back.bak'; 


加 消息 
已 为 数据 库 chapeer15'， 文 件 chapzer15，!( 位 于 文件 1 上 ;处 理 了 176 页 。 国 
已 为 数据 库 chapcer15'， 文 件 chapcer15_1og' (位 于 文件 1 上 ;处 理 了 z 页 。 
BACRUP DATABASE 成 功 处 理 了 178 页 ， 花 费 0.301 秒 (4.613 EB/ 秒 )。 


1 » 
回 查 .| wIDW-9527\M55QLSERYVER2008(..，| WIDW-9527\Administrato.,，| master | 00:00:00 | 0 行 
就 绪 行 4 列 1 chl Ins /4 


图 15.1 完整 备份 数据 库 chapter15 


1 
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从 图 15.1 所 示 的 效果 可 以 看 出 , 通过 BACKUP 语句 已 经 将 数据 库 chapter15 中 的 数据 
文件 和 日 志文 件 全 部 备份 到 了 chapter15_back.bak 文件 中 。 

【示例 2】 使 用 差异 备份 ， 备 份 数据 库 chapter15。 

根据 题目 要 求 ， 为 了 显示 差异 备份 的 效果 ， 这 里 仍然 将 chapter15 备份 到 文件 
chapter15_back.bak 中 。 具 体 的 语句 如 下 : 


BACKUP DATABASE chapter15 
TO DISK='D:\backdata\chapter15 back.bak' 
WITH DIFFERENTIAL; -- 差 异 备份 


执行 上 面 的 语句 ， 效 果 如 图 15.2 所 示 。 


本 Microsoft 5QL Server Management Studia [=] | 
文件 编辑 (E) 视图 W) 查询 (9) 项 目 P) 调式 0) 工具 (D 窗口 (W) 社区 (OO 帮助 (H) 
半 新 喜光 六 六 态 串 | 说 | 蕊 回电 | 冲 
a oa | master -| 执行 》 和 v 33 字 国 | 下 杷 | 的 圈 区 | 三 全 | 率 名 
5QLQuery1.sql -istrator (52))*| vx 
日 BACKUP DATABASE chapcerl5 
TO DISK='D:\backdata\chapter15 back.bak' 
WITH DIFFERENTIAL; ”-- 差 异 备份 


已 为 数据 库 “chapcer15'， 文 件 ' chapte et (位 于 文件 ? 上 ) 处 理 了 1 页 。 


BACKUP DATABAsSE WITH DIFPERENTIAL 成 功 钙 理 了 33 页 ， 花费 0.109 种 (z.30z 到 / 秒 )。 


已 为 数据 库 'chapterls'， 文件 'chapterls， (位 于 文件 ? 上 ) 处 理 了 32 页 。 - 


| WIDW-9527\MS5QLSERVER2008 (.,,| WIDW-9527VAdministrato.,，| master | 00:00:00 | 0 行 
行 6 列 1 chi Is 及 


图 15.2 差异 备份 数据 库 chapter15 


对 比 图 15.1 和 图 15.2 所 示 的 效果 可 以 看 出 ， 使 用 差异 备份 后 ， 备 份 的 数据 库 文件 要 
比 完整 备份 数据 库 时 的 文件 要 少 。 但 是 ， 这 两 种 备份 方式 ， 都 将 数据 库 的 数据 文件 和 日 志 
文件 进行 了 备份 。 在 实际 应 用 中 ， 针 对 同一 个 数据 库 最 好 采用 差异 备份 的 方式 ， 这 样 能 
大 大 节省 备份 时 间 。 


15.1.3 备份 日 志文 件 


在 上 一 小 节 中 已 经 讲解 了 数据 库 的 完整 备份 和 差异 备份 ， 但 是 这 两 种 备份 都 是 既 备 份 
数据 文件 又 备份 了 日 志文 件 。 如 果 数 据 库 管 理 员 只 需要 备份 日 志文 件 ， 就 需要 使 用 下 面 的 
语句 来 完成 了 。 

BACKUP LOG database 

TO DISK='path' 

其 中 : 

口 database: 要 备份 日 志文 件 的 数据 库 名 。 

口 path: 数据 库 备 份 的 目标 文件 ， 数 据 库 备 份 日 志文 件 的 扩展 名 是 .tmn。 例 如 : 备份 

到 Ci\data\a.trn。 

下 面 就 应 用 备份 数据 库 日 志文 件 的 语法 来 完成 示例 3 的 备份 操作 。 

【示例 3】 备份 chapter15 数据 库 中 的 日 志文 件 。 

根据 题目 要 求 ， 使 用 backup log 语句 即 可 完成 备份 。 具 体 语 句 如 下 : 


和 
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BACKUP LOG chapter15 
TO DISK=' D: \backdata\chapter1l5 backlog.trn' 


执行 上 面 的 语句 ， 效 果 如 图 15.3 所 示 。 


到 Microsol ft SQL Server Management Studio wr =|DI xl 
文件 (E) 编辑 (E) ”视图 (VW) 查阅 (9) 项 目 (p) 调式 (0) 工具 (D 窗口 (Ww) 社区 (QO 帮助 (H) 
卫 . 亲 查询 | 出 | 孔 韦 加 | 锯 | 记 加 本 | 台 掉 
Wh Wa | chapterl6 “| 执行 0_ hv 品名 国 | 下 榴 | 的 圈 扩 | 三 
‘SQL Query1.sql .istrator (52))* | ee 
日 BACKUP LOG chapter15 
入 DISK='D:\backdata\chapter15 backlog.trn'; 


消息 


已 为 数据 库 “chapcer15'， 文 件 chapcer15_ log，( 位 于 文件 ! 上 ) 处 理 了 1 页 。 
BACKUP L0G 成 功 处 理 了 1 页 ,花费 0.07s 秒 (0.013 te/ 秒 )。 


| 
园 查询 已 成 功 执行 .，| WIDW-9527WM55QLSERYER2008 (.,，| WIDW-9527\Administrato,,，| chapter16 | 00:00:00 | 0 行 
就绪 行 4 列 ! Chi Ins /4 


图 15.3 备份 数据 库 chapter15 的 日 志文 件 
从 图 15.3 所 示 的 效果 可 以 看 出 ， 通 过 上 面 的 语句 只 备份 了 数据 库 chapter15 中 的 日 志 


文件 。 
15.1.4 备份 文件 和 文件 组 


通过 前 面 内 容 的 学 习 ， 读 者 已 经 掌握 了 数据 库 的 基本 备份 方法 。 这 里 ， 将 继续 介绍 常 


用 备份 方式 的 最 后 一 种 方式 ， 即 文件 和 文件 组 的 备份 。 通 过 对 文件 和 文件 组 的 备份 ， 能 够 
更 快 地 恢复 数据 库 中 损坏 的 文件 。 备 份 语句 如 下 : 


BACKUP DATABASE database 

FILE='filename', 

FILEGROUP='groupname' 

TO DISK='path' 

其 中 : 

口 database: 要 备份 的 数据 库 名 。 

口 filename: 要 备份 数据 库 中 的 文件 名 。 这 里 ， 需 要 注意 的 是 文件 名 后 面 的 逗号 不 能 
够 省 略 。 

口 groupname: 要 备份 数据 库 中 的 文件 组 名 。 通 常 ， 数 据 库 默认 的 文件 组 是 主 文件 组 
primary 。 

口 path: 数据 库 备 份 的 目标 文件 ， 数 据 库 备 份 文件 的 扩展 名 是 .bak。 例 如 : 备份 到 
C:datava.bak。 

下 面 就 使 用 示例 4 来 完成 文件 和 文件 组 备份 的 操作 。 

【示例 4】 在 chapter15 中 ,创建 文件 组 group1， 并 在 其 中 添加 一 个 数据 文件 flel。 使 

BACKUP 语句 备份 group1 文件 组 下 的 数据 文件 filel。 
根据 题目 要 求 ， 创 建文 件 组 和 数据 文件 的 语句 如 下 : 


ALTER DATABASE chapter15 
ADD FILEGROUP groupl; =-- 添 加 文件 组 


本 汪汪 
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-添加 数据 文件 并 指定 文件 组 
ALTER DATABASE chapter15 
ADD FILE 
( 
NAME=filel, 
FILENAME='C: \ProgramFiles\MicrosoftSQOL 
Server\MSSQL10 .MSSQLSERVER2008\MSSQL\DATA\chapter15 filel.ndf' 


TO FILEGROUP groupl; 


执行 上 面 的 语句 ,， 即 可 为 chapter15 数据 库 中 添加 文件 组 和 文件 了 。 备份 文件 和 文件 组 
的 准备 工作 已 经 做 好 了 ， 下 面 就 通过 如 下 的 语句 来 对 其 进行 备份 了 。 


BACKUP DATABASE chapter15 

EFILE=' filel 

FILEGROUP='groupl' 

TO DISK=' D: \backqata\chapter15_ backfile.bak' 


执行 上 面 的 语句 ， 效 果 如 图 15.4 所 示 。 


[mcroson SOL Server Monagement Sd 
文件 E) 编辑 (视图 W 查询 (9) 项 目 P) 调试 D) 工具 (D 窗口 (Ww) 社区 (QO 帮助 (H) 

卫 和 建 查 电 W | 让 | 荔 也 四 | 记 | 区 回忆 | 到 忆 

ly Ha | master -| 1 执行 W 有 v 器 癌 国 | 玉 约 | 的 疼 拉 
SQLQueryl1.sql -...istrator (52))*| vx 
HBACKUP DATABASE chaprer15 

FILE='ftilel', 

FILEGROUP=' group1' 

TO DISK='D:\backdata\chapter15 backfile.bak' 


局 消息 
已 为 数据 库 'chapcer1ls'， 文件 ' tilel，( 位 于 文件 2 上 ) 处 理 了 s 页 。 a 
已 为 数据 库 'chapterls'， 文件 'chapter1l5_log (位 于 文件 2 上 ) 处 理 了 1 页 。 

BACKUP DATABASE.. .FILE=<nane> 成 功 处 理 了 9 页 ， 花 问 0.131 秒 t0.s33 Hp/ 秒 )。 

el 
园 查询 已 成 功 .。 | wIDw-9527\M55QLSERVER2008(..，| WIDW-9527\Administrato.,，| master | 00:00:00 | 0 行 
就 绪 行 7 列 1 chi ms 有 4 


图 15.4 备份 文件 和 文件 组 


从 图 15.4 所 示 的 效果 可 以 看 出 ， 已 经 将 新 创建 的 文件 备份 到 了 chapter15_backfile.bak 
文件 中 。 


15.1.5 ”使 用 企业 管理 器 备份 数据 库 

前 面 讲解 的 数据 库 备份 方法 ， 都 是 通过 SQL 语句 来 备份 的 。 使 用 SQL 语句 备份 数据 
库 时 ， 往 往 需要 记 住 很 多 不 同 的 语句 ， 对 于 初学 者 来 说 有 些 困难 。 但 是 ， 读 者 一 定 不 要 崩 
惧 ， 使 用 企业 管理 器 就 可 以 避免 了 牢记 备份 语法 的 工作 。 无 论 哪 种 备份 方法 ， 都 可 以 在 企 
业 管理 器 中 轻松 搞定 。 下 面 就 来 具体 看 看 在 企业 管理 器 中 是 如 何 备份 数据 库 的 。 通 过 如 下 
3 个 步骤 即 可 完成 备份 ， 这 里 仍然 以 备份 chapter15 数据 库 为 例 。 

1. 打开 数据 库 备份 界面 


在 “对 象 资源 管理 器 ”中 ， 展 开 “ 数 据 库 ”节点 ， 右 击 chapter15 数据 库 ， 在 弹出 的 右 
键 菜单 中 选择 “任务 ”|“ 备 份 ”选项 ， 效 果 如 图 15.5 所 示 。 


“314 。 
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用 备份 数据 库 - chapter15 


5 
选择 页 -也 和 
EE 
EE 可 
笋 二 ): sss 
恢复 模 式 中 - | 三 
备份 类 型 0); | 眉 3 司 
厂 仅 复制 备份 QD) 
备份 组 件 : 
数据 库 四) 
个 文件 和 文件 组 名 Pec | 
备份 集 


名 称 四 hterls5-< 芭 WE 人 
说 明 名 ) | 
备份 集 过 期 时 间 
个 胸 于 @): bp 3 
Ene ES -一 一 
目 
唤 呈 wwesusmmooe -a 


而 盘 四 个 型 凶 四 
连接 
TIDW-952T\Adninistrator 


对 查看 连接 属性 


就 绪 


图 15.5 数据 库 备 份 界面 
2. 选择 备份 类 型 及 备份 目标 


在 图 15.5 所 示 界 面 中 ， 选 择 备份 类 型 ， 这 里 选择 “完整 ”选择 备份 的 目标 时 可 以 通 
过 单 击 “ 添 加 ”按钮 ， 来 添加 备份 的 目标 位 置 ， 如 图 15.6 所 示 。 


前 过 择 各 份 目标 


选择 文件 或 备份 设备 作为 备份 目标 。 悠 可 以 为 常用 文件 包 陵 备份 设备 。 


图 15.6 添加 备份 目标 位 置 


在 图 15.6 所 示 界 面 中 ,选择 备份 的 位 置 ， 并 单 击 “ 确 定 ”按钮 ， 即 可 完成 备份 位 置 的 


3. 确认 完成 
在 图 15.5 所 示 界 面 中 ,填写 完 相应 的 备份 信息 后 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 数据 


“ss 
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库 的 备份 操作 。 效 果 如 图 15.7 所 示 。 


Microsol ft SQL Server Management Studio xl 


Wy 对 数据 库 "chapter15 的 备份 已 成 功 完成 。 


中 [| 


图 15.7 备份 操作 完成 提示 


通过 上 面 的 3 个 步骤 ， 就 完成 了 一 个 完整 的 备份 数据 库 的 操作 ， 对 于 其 他 类 型 的 备份 
操作 ， 读 者 可 以 自己 尝试 完成 。 


15.2 还原 数 据 库 


还 原 数据 库 也 称 为 恢复 数据 库 ， 都 是 对 备份 数据 库 文件 进行 操作 的 。 如 果 数 据 库 不 能 
够 进行 还 原 操作 ， 那 么 ， 备 份 数 据 库 也 就 失去 意义 了 。 在 本 节 中 ， 主 要 讲解 如 何 将 备份 后 
的 数据 文件 还 原 。 


15.2.1 还 原 数据 库 文件 


还 原 数据 库 操作 是 否 能 完整 ， 主 要 取决 于 备份 数据 库 的 文件 。 在 备份 数据 库 时 ， 备 份 
方式 主要 有 备份 数据 库 、 备 份 数 据 库 文件 以 及 备份 文件 组 及 文件 的 方式 。 下 面 首先 来 为 读 
者 讲解 如 何 还 原 备份 的 数据 库 。 具 体 的 语法 形式 如 下 : 

RESTORE DATABASE database 

FROM DISK= 'path'; 

其 中 : 

口 database: 要 还 原 的 数据 库 名 。 

口 path: 数据 库 的 备份 文件 路 径 。 

下 面 就 使 用 示例 5 来 演示 如 何 还 原 数 据 库 。 

【示例 S】 将 chapter15 数据 库 删 除 ， 并 使 用 数据 库 备份 文件 还 原 数据 库 chapter15。 

根据 题目 要 求 ， 首 先 要 删除 数据 库 chapter15， 语 句 如 下 : 


DROP DATABASE chapterl15; 


通过 上 面 的 语句 ， 即 可 将 数据 库 chapter15 删除 了 。 然 后， 通过 RESTORE 语句 将 数据 
库 还 原 ， 具 体 语 句 如 下 : 

RESTORE DATABASE chapter15 

FROM DISK='D:\backdata\chapter15 back.bak'; 

执行 上 面 的 语句 ， 效 果 如 图 15.8 所 示 。 

从 图 15.8 所 示 的 效果 可 以 看 出 ， 已 经 通过 备份 文件 chapter15_back.bak 将 数据 库 
chapter15 还 原 了 。 
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x Microsof ft SQL Server nt Studio 
文件 (E) ”编辑 E) ”视图 (WW 坦 渔 (0) 项 目 (6) 调 江 (0) 工具 (D 窗 D(W 社区 (CO 帮助 由 
:也 新建 查询 岂 六 | 壤 访 十 | 访 | 芒 回忆 | 加 

a 2 | master -| ?执行 WW 》 v 中 时 国 | 3 绚 | 的 了 央 


Souery i sa istrator C2) a 
日 RESTORE DATABASE chapter15 
上 FRON DISK='D:\backdata\ chapter 15 back.bak'; 


Managemei 


已 为 数据 库 “chaprer15'， 文 件 ee rls， (位 于 文件 1 上 ) 处 理 了 176 
已 ; 库 'chapverls'， 文 件 ， rl5_log， (位 于 文件 1 上 ;处 理 了 z 页 。 
至 sTopE DATABAsE 成 功 处 理 了 i 3 0-173 秒 t8.027 亚 /种 )。 


ul 
回 查询 已 | wIDw-9527\M55QLSERYER2008(,.，| WIDW-9527VAdministrato,.，| master | 00:00:01 | 0 行 
就 绪 行 4 列 1 chl ms /4 


图 15.8 ”还原 数 据 库 chapter15 


15.2.2 还 原文 件 和 文件 组 


在 备份 数据 库 时 ， 可 以 直接 备份 数据 库 文件 或 文件 组 ;在 还 原 数据 库 时 ， 也 可 以 将 文 
件 组 和 文件 还 原 。 具 体 的 语句 如 下 : 


RESTORE DATABASE database 
FILEGROUP|FILE=' filename 
FROM DISK='path' 

[WITH REPLACE]; 


其 中 : 

口 database: 要 还 原 的 数据 库 名 。 

口 filename: 要 还 原 数据 库 中 的 文件 名 或 文件 组 名 。 
口 path: 文件 或 文件 组 的 备份 路 径 。 

口 WITH REPLACE: 替换 原 有 的 文件 组 。 

下 面 就 通过 示例 6 来 演示 如 何 还 原文 件 和 文件 组 。 
【示例 6】 还 原 在 示例 4 中 备份 的 文件 组 group1。 
根据 题目 要 求 ， 还 原 的 具体 语句 如 下 : 


RESTORE DATABASE chapter15 

FILEGROUP ='groupl' 

FROM DISK=' D: \backdata\chapter1l5 backfile.bak' 

WITH REPLACE; 

执行 上 面 的 语句 ， 效 果 如 图 15.9 所 示 。 

从 图 15.9 所 示 界 面 中 ， 就 可 以 看 出 文件 组 groupl 已 经 被 还 原 了 。 


全 注意 ; 如 果 上 面 的 语句 ， 不 能 正常 执行 ， 请 先 将 数据 库 chapter15 设置 成 脱 机 状态 。 设 
置 成 脱 机 状态 的 方法 是 右 击 chapter15 数据 库 ， 在 弹出 的 右键 菜单 中 选择 “任务 ” 
|“ 脱 机 ”选项 ， 弹 出 图 15.10 所 示 界 面 。 

在 图 15.10 所 示 界 面 中 ， 单 击 “ 关 闭 ” 按 钮 ， 即 可 完成 数据 库 脱 机 状态 的 设置 。 
相反 ， 如 果 完 成 了 还 原 操作 ， 可 以 右 击 chapter15 数据 库 ， 在 弹出 的 右键 菜单 中 
选择 “任务 ”|“ 联 机 ”选项 ， 即 可 将 数据 库 设 置 成 联机 状态 。 


ars 
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[wmeoosoLsevermonaoments LI 
文件 EE) 编辑 (E) 视图 (由 ”查询 (9) 项目 P) 调试 D) 工具 (D 窗口 (Ww) 社区 (QO 帮助 (H) 
也 新 寻 查询 | 让 | 了 也 辐 | 记 | 蕊 加 己 双 天 
时 2 | master ~| 执 和 5 bvV 纪 训 
SQLQuery1.sql -istrator (52))*| 


| 
2a 丁 加 的 | 三 皇 | 棕 训 
司 | ?” 框 | 怒 圈 引 三 全 Ei 
日 RESTORE DATABASE chapter15 
FILEGROUP ='groupl' 
FROM DISk='D:\hackdata\ chapter15 backrile.bak 
WITH REPLACE; 


局 
E 
4 上 生 
四 消息 
已 为 数据 库 'chapterls'， 文 件 ' tilel，( 位 于 文件 1 上 ) 处 理 了 8 页 。 
已 为 数据 库 “chapcer15'， 文 件 ,chapcer15_log' (位 于 文件 1 上 ) 处 理 了 z 页 。 
由 于 文件 ' chapcerls， 当前 止 还 原 到 Lsw 39000000014000001， 但 其 必 : 
前 深 开 始点 现在 位 于 日 志 序 列 号 (Lsw) 
此 REsTons 语句 成 功 地 执行 了 一 些 操作 ， 
ESTOPE DATABASE .-. 


39000000014000001 处 。 需 


其 必须 还 原 到 5sw 39000000014000001， 因 此 无 法 使 该 数据 库 联机 。 | 
要 继续 前 党 到 zsH 39000000014000001 能 完成 还 原 顺 序 。 
， 但 由 于 需要 一 个 或 多 个 REsTons 步 又 ， 无 法 使 数据 库 在 线 。 以 前 的 消息 说 明了 此 时 无 法 进行 恢复 的 原因 。 
FILE=<name> 成 功 处 理 了 10 页 ， 花费 0.061 秒 (1.184 到/ 秒 )。 = 
= 一 =| :者 
回 查询 已 成 功 执行 。 
就 绪 


| WIDW-9527\M55QLSERVER2008 (..，| WIDW-9527VAdministrato..，| master | 00:00:00 | 0 行 
行 7 


列 1 
图 15.9 还 原文 件 组 


Chl Ins 阿 
者 使 数据 库 脱 机 - WIDW-9527\MSSQLSERYER2008 
@w 


=j9jx| 
1 总 计 
1 成 功 
详细 信息 中 ) 


0 错误 
0 警 肯 


使 数据 库 “chapter15” 脱 机 


图 15.10 设置 数据 库 脱 机 
15.2.3 ”使 用 企业 管理 器 还 原 数据 库 


还 原 数据 库 既 可 以 使 用 SQL 语句 来 完成 , 也 可 以 使 用 企业 管理 器 来 完成 。 使 用 企业 管 
理 器 来 还 原 数据 库 是 更 为 简单 的 一 种 方式 。 但 是 ， 无 论 使 用 哪 种 方式 来 还 原 数 据 库 都 要 明 
确 备份 数据 库 的 位 置 才 可 以 哦 ! 因此 ， 一 定 要 小 心 存放 备份 的 信息 。 使 用 企业 管理 器 还 原 
数据 库 通常 需要 如 下 4 个 步骤 即 可 。 


1. 打开 还 原 数据 库 界面 


在 “对 象 资源 管理 器 ”中 ， 右 击 “ 数 据 库 ”选项 ， 在 弹出 的 右键 菜单 中 选择 “还 原 数 
据 库 ”选项 ， 弹 出 如 图 15.11 所 示 界 面 。 


2. 指定 备份 的 位 置 
在 图 15.11 所 示 界 面 中 ， 输 入 目标 数据 库 的 名 称 或 者 直接 在 下 拉 列 表 中 选择 一 个 数据 
库 名 称 ， 单 击 “ 源 设备 ”后 面 的 天 | 按钮 ， 弹 出 图 15.12 所 示 界 面 。 


在 图 15.12 所 示 界 面 中 ， 单 击 “ 添 加 ”按钮 ， 找 到 备份 数据 库 的 位 置 ， 单 击 “ 确 定 ” 
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名 称 | 组 件 | 类 型 | 服务 器 | 数据 库 | 位 置 


EESIEEENGEECESIIEETSTE 


服务 器 
TDY-952TWISSQLSERYER2008 


连接 
WIDN-9527\Adninistrator 


对 查看 连接 属性 


就绪 


图 15.11 还 原 数据 库 界面 


到 
指定 还 原 操作 的 备份 媒体 及 其 位 置 . 
备份 媒体 @) EF “| 
备份 位 置 LL); 


图 15.12 ”指定 备份 位 置 


按钮 即 可 。 如 图 15.13 所 示 。 
3. 选择 用 于 还 原 的 备份 集 


在 图 15.13 所 示 界 面 中 ， 列 出 了 备份 文件 中 所 有 备份 过 的 文件 信息 。 在 显示 的 所 有 备 
份 集中 ， 选 择 要 还 原 的 备份 集 。 可 以 同时 选择 多 个 备份 集 的 信息 。 选 择 完成 后 ， 单 击 “ 确 
定 ” 按 钮 ， 弹 出 图 15.14 所 示 界 面 。 
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击 还 原委 大 库 - chapterls 


和 择 页 
EE 
村 选项 

3 数据 库 完整 YIDW-9527\RSSQLSERYER2008 chapterl5 39000000005600064 

口 数据 库 莽 异 WIDW-9527\MSSQLSERVER2008 chspterl5 2 39000000009200034 3900000001! 
口 数据 库 ”差异 ”TDI-952TVISSQLSERYER2008 chapterl5 3 39000000010700001 3900000001! 
服务 器 口 数据 库 ”差异 YIIW-9527WISSQLSERYER2006 chapterl5 4 39000000010900001 。 3900000001 
TY-9527VISSQLSERYEF2008 [a] 数据 库 完整、WIDW-9527\RSSQLSERVER2008 chapterl5 5 39000000011300037 3900000001: 
连接 rc 数据 库 莽 异 WIDW-9527\MSSQLSERVER2008 chapterl5 6 39000000013600034 ”3900000001! 
WIDN-9527\Adninistr ator [eo 获 异 WIDW-9527\NSSQLSERVER2008 chapterl5 7 39000000015100001 3900000001! 
于。 查看 连接 属性 


图 15.14 还 原 chapter15 数据 库 成 功 的 提示 


在 图 15.14 所 示 界 面 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 完成 数据 库 chapter15 的 还 原 操作 。 
在 企业 管理 器 中 ， 也 可 以 还 原 日 志文 件 以 及 文件 或 文件 组 ， 操 作 方 法 都 与 还 原 数据 库 的 方 
法 类 似 ， 在 这 里 就 不 一 一 列举 了 。 


15.3 ”数据 库 的 分 离 和 附加 


数据 库 的 备份 和 还 原 是 一 种 数据 库 移植 的 好 方法 ， 同 时 也 是 数据 库 管 理 员 首选 的 数据 
库 管理 方式 之 一 。 除 了 数据 库 的 备份 和 还 原 之 外 ， 还 有 一 种 更 为 简便 的 方式 ， 就 是 数据 库 
的 分 离 和 附加 。 分 离 数 据 库 是 完整 地 保存 了 数据 库 的 数据 文件 和 日 志文 件 ， 附 加 数据 库 则 
不 需要 重新 创建 数据 库 ， 直 接 将 分 离 后 的 文件 附加 即 可 。 


15.3.1 数据 库 的 分 离 
所 谓 数 据 库 的 分 离 ， 就 是 在 当前 连接 的 数据 库 中 将 某 个 数据 库 文件 去 除数 据 连接 ， 并 
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能 够 独立 地 复制 到 其 他 的 计算 机 中 。 数 据 库 的 分 离 通常 是 使 用 企业 管理 器 直接 完成 的 ， 也 
可 以 使 用 系统 存储 过 程 SP_DETACH DB 来 完成 。 下 面 就 通过 示例 7 和 示例 8 分 别 演 示 如 
何 使 用 企业 管理 器 和 系统 存储 过 程 来 分 离 数 据 库 。 

【示例 7】 使 用 企业 管理 器 分 离 数据 库 chapter15 。 

在 企业 管理 器 中 ， 分 离 数据 库 chapter15， 需 要 如 下 两 个 步骤 即 可 完成 。 

(1) 打开 分 离 数 据 库 界 面 

在 “对 象 资源 管理 器 ”中 ， 展开“ 数据 库 ” 节 点 ， 右 击 chapter15 数据 库 ， 并 在 弹出 的 
右键 菜单 中 选择 “任务 ”|“ 分 离 ” 选 项 ， 弹 出 如 图 15.15 所 示 界 面 。 

者 分 享 煞 气 库 = 上 jx 


选择 页 
加 


服务 器 
WIDW-9527\MSSQLSERVER2008 


连接 
WIDN-9527\Adninistrator 


圣 查看 连接 属性 


图 15.15 分离 数据 库 界 面 


(2) 确认 分 离 

在 图 15.15 所 示 界 面 中 ， 可 以 看 到 在 数据 库 chapter15 后 面 有 两 个 复 选 框 ， 一 个 是 删除 
连接 的 ， 一 个 是 更 新 统计 信息 的 。 其 中 ， 如 果 选 中 “删除 连接 ”选项 ， 可 以 断 开 所 有 与 该 
数据 库 相 关 的 连接 。 这 里 ， 选 中 “删除 连接 ”选项 ， 并 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 数据 
库 的 分 离 操作 。 数 据 库 分 离 后 ， 数 据 库 在 对 象 资源 管理 器 中 就 不 存在 了 。 此 时 ， 分 离 后 的 
数据 库 就 可 以 在 创建 数据 库 的 目录 下 随意 地 移动 位 置 了 。 

【示例 8】 使 用 系统 存储 过 程 SP_DETACH DB 来 完成 数据 库 chapter15 的 分 离 操作 。 

根据 题目 要 求 ， 分 离 数 据 库 的 语句 如 下 : 


SP_DETACH DB @dbname="'chapter15'; 
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了 Microsoft SQL Server Management Studig 


执行 上 面 的 语句， 效果 如 图 15.16 所 示 。 
在 图 15.16 所 示 界 面 中 , 可 以 看 出 chapter15 已， exo eso 
经 被 成 功 分 离 了 。 读 者 可 以 在 对 象 资源 管理 器 中 刷 2 记名 BBD 


新 数据 库 节 点 , 看 看 数据 库 chapterl5 是 还 存在 ?erm 


日 SP_DETACH_DB Bdbname= 


15.3.2 ”数据 库 的 附加 下 = 
J si 


与 数据 库 的 分 离 相对 应 ， 数 据 库 的 附加 也 可 以 pee [wwe [mae veron |e 
通过 两 种 方式 来 实现 。 一 种 是 使 用 企业 管理 器 来 完 
成 ， 一 种 是 通过 系统 存储 过 程 来 完成 。 下 面 就 分 别 
通过 示例 9 和 示例 10 来 完成 数据 库 附加 的 操作 。 

【示例 9】 使 用 企业 管理 器 完成 对 chapter15 的 附加 操作 。 

在 企业 管理 器 中 ， 附 加 数据 库 chapter15 通过 以 下 3 个 步 马 即 可 完成 。 

(1) 打开 附加 数据 库 界 而 

在 “对 象 资源 管理 器 ”中 ， 右 击 “ 数 据 库 ”节点 ， 在 弹出 的 右键 菜单 中 选择 “附加 ” 
选项 ， 弹 出 如 图 15.17 所 示 界面 。 


图 15.16 ”分离 数 据 库 chapter15 


=lojxl 
nt -Dw 
EE 
要 附加 的 数据 库 D) 
MDF 文 件 位 置 数据 库 名 称 附加 为 所 有 者 状态 消息 
数据 库 详 细 信 息 CD 
原 她 文件 名 文件 类 到 当前 文件 路 径 消息 
服务 器 
es 
连接 
对 ”查看 连接 悍 性 
进度 
pe 下 加 目录 人 0) 和 只 QD) 


图 15.17 附加 数据 库 界 面 


(2) 选择 需要 附加 的 数据 库 
在 图 15.17 所 示 界 面 中 ， 单 击 “ 添 加 ”按钮 ， 来 添加 需要 附加 的 数据 库 。 如 图 15.18 


所 示 。 
在 图 15.18 所 示 界 面 中 ， 单 击 要 附加 的 数据 库 名 称 ， 并 单 击 “确定 ”按钮 ， 即 可 完成 
附加 数据 库 的 选择 。 这 里 ， 选 择 chapter15 数据 库 ， 效 果 如 图 15.19 所 示 。 
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定位 数据 库 文 件 - 527\MSSQLSERYER2008 二 回避 


选择 文件 @): 


日 Microsoft SQL Server 
外国 100 
外国 上 
由 例 % 
-a NSAS10. NSSQLSERYER2008 
由 国 MSRS10. NSSQLSERYER2008 
由 国 wssqL 1 
BG nssqL10. NSSQLSERYER2008 
nssor 
由 国 Bacp 
由 全 Bim 
日 DTA 
DD chapterl0. ndf 
DD chapterll.ndf 
DD chapterl2. ndf 
DD chapterl3.ndf 
DD chapterl4 ndf 


DD chapter3. 
DD chapter4. ndf 

DD chapters.ndf 

DD chapter6. ndf 

DD chapterT. ndf 

DD chapter8.naf 

DD chapterg .mdf 

DD waster ndf 

DD nodel ndf 

DD wsnppsta ndf 

口 ReportServer$MSSQLSERVER2008. mdf 

DD ReportServer$MSSQLSERVER2006TenpDB. ndf 

DD tenpdb.ndf | 


所 过路 径 到) Eeew FilesWierosoft SL ServerWISS9LIO IESS 
文件 类 型 TD) ET 人 .mdf) 局 
文件 名 四: Jastels mdf 


CE | 


前 阳 加 数据 库 


服务 器 
WIDY-9527 ‘MSSQLSERVER2008 


连接 
WIDE-9527\Adninistrator 


竺 查看 连 近 民 考 


Ps 天 加 和 人 Ea 


图 15.18 选择 要 附加 的 数据 


“chapter15” 数 据 库 详细 信息 ID) 


CAProerm Files\Microsoft 


连接 | seteisoae | 
FE rr 上 Croee Fileauiererot 


shapter15_filel ndf 数据 Ci\Progre FilesVliereseft 


[eal 


图 15.19 选择 附加 数据 库 后 的 效果 
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(3) 确认 附加 

在 图 15.19 所 示 界 面 中 ， 单 击 “ 确 定 ”按钮 ， 即 可 完成 数据 库 chapter15 的 附加 操作 。 
【示例 10】 使 用 系统 存储 过 程 SP_ATTACH DB 完成 数据 库 chapter15 的 附加 。 
根据 题目 要 求 ， 附 加 语句 如 下 : 


SP ATTACH DB edbname='chapter15'， 
@filenamel='C:\ProgramFiles\MicrosoftSQL 
Server\MSSQL10 .MSSQLSERVER2008\MSSQL\DATA\chapter15.mdf'; 


执行 上 面 的 语句 ， 效 果 如 图 15.20 所 示 。 


> Microsol ft SQL Ser' 
文件 ”编辑 (E) 
EE 

;于 骨 | master "|! 》 mv 时 晤 国政 续 | 友 轩 太 | 三 | 素来 | 多 


nt Studio 
项 目 B) 调式 0) 工具 (D 窗 D(w) 社区 (QO 帮助 td 


“5QLQuery1.sql -.-istrator (52))*| vx 
日 sp_ATTACH_DB Bdbname='chapter15' 
上 BFILENAME1-'C:VProgram Files 


crosoft SQL Server\MSSQL10.NSSQLSERVER2008\ NSSQL\DATA\chapter15,mdf' ; 


» 
回 查询 已 成 功 执行 | WIDW-9527\MS5QLSERVER2008 (… | WIDW-9527\Administrato,.，| master | 00:00:00 | 0 行 


行 5 


图 15.20 ”使 用 存储 过 程 附加 数据 库 chapter15 


列 1 Chl ms 有 4 


通过 图 15.20 所 示 的 效果 ， 可 以 看 出 数据 库 chapter15 已 经 被 附加 了 。 
全 注意 ; 在 使 用 SP_ATTACH DB 系统 存储 过 程 对 数据 库 进行 附加 时 ，@dbname 与 


@filenamel 语句 之 间 要 用 去 号 隔 开 。 此外, 一 次 可 以 附加 多 个 数据 文件 ， 最 多 可 
以 附加 16 个 数据 文件 。 如 果 要 附加 多 个 数据 文件 ， 则 只 需要 在 @filenamel 后 面 
用 过 号 隔 开 继续 写 @filename2 即 可 ， 依 此 类 推 。 


15.4 本 章 小 结 


本 章 主要 讲解 了 数据 库 的 备份 和 还 原 以 及 数据 库 的 分 离 和 附加 。 在 数据 库 的 备份 和 还 
原 部 分 主要 讲解 了 数据 库 的 备份 类 型 、 常 用 的 备份 语句 、 还 原 语句 以 及 如 何 使 用 企业 管理 
器 对 数据 库 进 行 备份 和 还 原 ， 在 数据 库 的 分 离 和 附加 部 分 ， 主 要 讲解 了 使 用 系统 存储 过 程 
和 企业 管理 器 来 分 离 和 附加 数据 库 。 通 过 本 章 的 学 习 ， 读 者 可 以 对 自己 创建 的 数据 库 进 行 
备份 ， 以 便 今 后 丢失 时 可 以 恢复 。 


15.5 本 章 习题 


一 、 填 空 题 

1. 在 SQL Server 中 ， 常 见 的 备份 方式 有 和 3 种 。 
2. 在 SQL Server 中 ， 备 份 数据 库 使 用 的 关键 字 是 。 

3. 在 SQL Server 中 ， 还 原 数据 库 使 用 的 关键 字 是 。 
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二 、 选 择 题 
1. 分 离 数据 库 时 使 用 的 系统 存储 过 程 是 8 
A. SP RENMAE B. SP_ATTACH 
C. SP_DETACH DB D. 以 上 都 不 是 
2. 还 原 数 据 库 chapter15 所 使 用 的 语句 正确 的 是 。 


A. RESTORE DATABASE chapter 15 TO 'c:\data'; 
B. RESTORE DATABASE chapter 15 FROM 'c:\data'; 
C. RESTORE DATABASE chapter 15 FROM 'c:\data\chapter15.bak ; 
D. 以 上 都 不 对 
3. 使 用 系统 存储 过 程 附加 数据 库 chapter15 的 语句 正确 的 是 
A. SP DETACH DB @filename= 'c:\chapterl$.mdf’; 
B. SP_ATTACH DB @dbname= 'chapterl5',@filenamel= 'c:\chapterl5.mdf’; 
C. SP_ATTACH DB @dbname= 'chapterl15'; 
D. 以 上 都 不 对 


三 、 问 答题 


1. 使 用 企业 管理 器 还 原 数据 库 的 步骤 是 什么 ? 
2. 差异 备份 与 完整 备份 有 什么 区 别 ? 
3. 分 离 数 据 库 与 备份 数据 库 有 什么 区 别 ? 


、 操 作 题 


创建 名 为 chapter15_1 的 数据 库 ， 并 完成 如 下 操作 。 
(1) 备份 chapter15_1 数据 库 。 

(2) 备份 chapter15_1 数据 库 的 日 志文 件 。 

(3) 还 原 chapter15_1 数据 库 。 

(4) 使 用 系统 存储 过 程 分 离 数 据 库 chapter15_1。 
(5) 使 用 系统 存储 过 程 附加 数据 库 chapter15_1。 
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在 SQL Server 中 , 系统 管理 员 不 仅 可 以 以 手动 的 方式 来 管理 数据 库 , 也 可 以 通过 系统 
提供 的 多 种 自动 化 方式 辅助 管理 数据 库 。 这 些 自动 化 方式 主要 包括 作业 、 维 护 计划 、 和 警报 
以 及 操作 员 等 。 在 本 章 中 就 将 带领 读者 一 一 认识 并 使 用 它们 。 

本 章 的 主要 知识 点 如 下 : 

如 何 使 用 SQL Server 代理 
如 何 使 用 作业 

如 何 使 用 计划 

如 何 使 用 警报 

如 何 使 用 操作 员 


DOOOO 


16.1 SQL Server 代理 


SQL Server 代理 是 用 来 完成 所 有 自动 化 任务 的 重要 组 成 部 分 ， 可 以 说 ， 所 有 的 自动 化 
任务 都 是 通过 SQL Server 代理 来 完成 的 。 在 本 节 中 , 就 将 学 习 SQL Server 代理 的 用 途 及 使 
用 方法 。 

16.1.1 认识 SQL Server 代理 


所 谓 SQL Server 代理 ， 就 是 代替 用 户 去 完成 一 系列 的 操作 。 这 些 操作 就 是 后 面 要 学 习 
的 作业 、 和 警报 以 及 计划 等 .实际 上 ,SQL Server 代理 就 是 一 种 服务 ,服务 的 名 称 是 SQL Server 
Agent。 下 面 就 看 看 在 什么 地 方 能 够 看 到 这 个 服务 吧 。 如 果 你 在 安装 SQL Server 时 ， 没 有 
选择 开机 自动 启动 服务 选项 ， 那 么 ， 每 次 你 都 需要 在 Windows 资源 管理 器 中 的 服务 中 ， 启 
动 SQL Server 服务 。 没 错 ，SQL Server 代理 服务 也 是 在 这 个 页 面 启动 的 。 在 “开始 ”菜单 
下 的 “设置 ”|“ 控 制 面板 ”界面 ， 单 击 “ 管 理工 具 ” 选 项 ， 然 后 单 击 “ 服 务 ” 选 项 ， 即 可 
看 到 SQL Server 的 代理 服务 了 ， 如 图 16.1 所 示 。 

从 图 16.1 所 示 界 面 中 , 可 以 看 到 SQL Server 代理 服务 没有 启动 。 在 下 一 小 节 中 ， 就 将 
讲解 如 何 启动 、 设 置 以 及 停止 SQL Server 代理 。 


16.1.2 操作 SQL Server 代理 


SQL Server 代理 承载 着 一 系列 的 自动 化 任务 ， 那 么 ， 使 用 好 SQL Server 代理 就 尤为 重 
要 了 。 下面 就 从 SQL Server 代理 的 设置 、 启 动 以 及 停止 3 个 操作 来 讲解 SQL Server 代理 的 
使 用 。 
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3 =I9lxl 
文件 E) ”操作 (A) ”查看 VW) 才 助 (中 
如 清国 | 轿 加 姑 | 加 国 | 9 
服务 (二 地 ) 区 区 [3 
5QL Server (SQLEXPRESS) 提供 ..， 已 启动 ”自动 网 络 服务 
钨 5QL 5erver Analysis Services (M55Q rbd 已 启动 自动 网 络 服 务 
人 5QL 5erver browser ,.， 已 启动 自动 网 络 服务 
雹 5QL 5erver Integration Services 10.0 > ,， 已 启动 自动 网 络 服务 
hy 5QL 5erver Reporting Services (M55.,. 人 .， 已 启动 自动 网 络 服务 
多 5QL5ervery55wrter 提供 … 已 启动 自动 本 地 系统 
SQL 5erver 代理 (M55QL5ERYER2006) 其 - 手动 | 
SQL Server 代理 (SQLEXPRESS) 我 行 .… 票 用 
5tarWind AE 5ervice Enab..， 已 启动 自动 本 地 系统 
钨 system Event Noufication 监视 .。 已 启动 ”自动 本 地 系统 
信 y5ystem Update 已 启动 自动 本 地 系统 
SB Task Scheduler 使 用 ,.， 禁用 本 地 系统 | 
纪 yTCP/IP NetBIOS Helper 提供 ..， 已 启动 ”自动 本 地 服务 
纺 Telephony 提供 ..， 已 启动 手动 本 地 系统 
二 Tenet 区 许 .… 禁用 本 地 服务 
i Tencent 5050 Update Service 已 启动 ”自动 本 地 系统 
钨 Temnal Services 区 许 .， 已 局 动 手动 本 地 系统 | 
\ 扩 展 入 标准 


图 16.1 SQL Server 代理 服务 


1. 设置 SQL Server 代理 


在 图 16.1 所 示 界 面 中 ， 右 击 “SQL Server 代理 ”选项 ， 在 弹出 的 右键 菜单 中 选择 “ 属 
性 ”选项 ， 弹 出 如 图 16.2 所 示 的 界面 。 

在 此 界面 中 ， 可 以 看 到 SQL Server 代理 的 基本 人 信息。 其中， 在 “登录 ”选项 卡 界面 还 
可 以 为 该 服务 设置 登录 账户 ， 如 图 16.3 所 示 。 


本 了 5 sar server 代理 (M55QLSERYER2009) 的 尾 性 (本 二 计 竺 可 | 
人 常规 | 登录 | 恢复 | 依存 关系 | 常规 。 登录 “| 估 复 | 依存 关系 | 
服务 名 称 : SQLAgent$MSSQLSERVER2008 车 录 身份 : 
ssi 0 SE CS 
= TRTTY VETORSERT 济 顷 @@) 
可 执行 文件 申 径 0 BD: FE 
FNProgan Leer SL Sarvar ISLi0 SLR 确 AEC): [as 
-I 习 悠 可 局 用 或 茜 用 以 下 所 列 的 硬件 也 轩 文件 服务 ); 
服务 状态 已 停止 Yndocked ee 


局 动 G) 停 IED 看 停 回 [ 肥 池 司 


当 从 此 处 启动 服务 时 ， 您 可 指定 所 适用 的 启动 参数 。 


司 动 # 数 m 「[ 后 用 E) | 禁用 四 ) 


取消 “| 应用 上 取消 碳 用 [8 
图 16.2 SQL Server 代理 属性 界面 图 16.3 设置 登录 账户 


在 图 16.3 所 示 界 面 中 ， 可 以 设置 不 同 的 登录 账户 。 其 中 ， 本 地 系统 账户 就 是 指 内 置 的 
本 地 系统 管理 员 账户 ;此 账户 是 指 运行 SQL Server 代理 服务 的 Windows 域 账户 。 也 可 以 
通过 单 击 “ 浏 览 ”按钮 ， 重 新 选择 域 账 户 。 


外 说 明 : 如 果 在 图 16.2 所 示 界 面 中 ， 将 SQL Server 代理 的 启动 类 型 更 改 成 自动 ， 则 在 计 
算 机 启动 时 ， 就 会 自动 启动 该 服务 了 。 但 是 ， 还 是 建议 读者 将 其 设置 成 “手动 ” 
方式 ， 这 样 能 够 节省 计算 机 的 开机 时 间 。 
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2. 启动 SQL Server 代理 


启动 SQL Server 代理 服务 很 简单 ， 与 启动 其 他 服务 的 操作 是 一 样 的 。 只 需要 在 图 16.1 
所 示 界 面 中 ， 右 击 “SQL Server 代理 ”， 在 弹出 的 右键 菜单 中 选择 “启动 ”选项 ， 弹 出 如 
图 16.4 所 示 的 界面 。 


x 
Windows 正在 尝试 启动 本 地 计算 机 上 的 下 列 服务 
SQL Server 代理 SSQLSERVER2008) 


图 16.4 启动 SQL Server 代理 


当 图 16.4 所 示 界 面 中 的 进度 条 走 到 头 ， 就 可 以 完成 SQL Server 代理 服务 的 启动 操作 。 另 
外 ， 也 可 以 通过 在 图 16.2 所 示 界 面 中 ， 直 接 单 击 “ 启 动 ” 按 钮 ， 启 动 SQL Server 代理 服务 。 

3. 停止 SQL Server 代理 

停止 SQL Server 代理 的 操作 与 启动 SQL Server 代理 的 操作 类 似 , 一 种 方式 是 在 图 16.1 
所 示 界 面 中 ， 右 击 “SQL Server 代理 ”， 在 弹出 的 右键 菜单 中 选择 “停止 ”选项 ， 即 可 停止 
该 服务 ， 另 一 种 方式 是 在 图 16.2 所 示 界 面 中 ， 单 击 “ 停 止 ”按钮 ， 也 可 以 停止 该 服务 。 


16.2 作 业 


作业 可 以 看 作 是 一 个 任务 ， 在 SQL Server 代理 中 使 用 最 多 的 就 是 作业 了 。 每 一 个 作业 
都 是 一 个 或 多 个 步骤 组 成 的 ， 有 序 地 安排 好 每 一 个 作业 步骤 ， 就 能 够 有 效 地 使 用 作业 了 。 
在 本 节 中 就 将 带领 读者 来 学 习 如 何 创建 和 使 用 作业 。 


16.2.1 创建 作业 


在 SQL Server 中 ,创建 作业 通常 都 是 借助 企业 管理 器 来 完成 的 。 通常 创建 作业 分 为 如 
下 两 个 步骤 。 


1. 打开 创建 作业 的 界面 


在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ”节点 ， 右 击 “ 作 业 ” 节 点， 在 弹 
出 的 快捷 菜单 中 选择 “新 建 作业 ”命令 ， 弹 出 界面 如 图 16.5 所 示 。 


2. 添加 作业 信息 


在 图 16.5 所 示 界 面 中 ， 填 入 作业 名 称 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 作业 的 创建 操作 。 但 

是 ， 此 时 创建 的 作业 没有 任何 功能 。 如 果 要 让 该 作业 完成 一 些 功能 ， 就 必须 要 为 作业 添加 步骤 。 

全 注意 : 如 果 不 想 作业 创建 后 就 马上 执行 ,那么 需要 在 “新 建 作业 ”界面 中 清除 “已 启用 ” 
复 选 框 的 选中 状态 。 
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Ed| 
Su -DB 
空当 
字 步 村 名 称 加 : mv 
至 计划 
他 警报 iistrater 
名 癌 所 有 者 四 ) Fors mini 天 | 
闻 旧 标 OO: [ERE || 
说 明 O) 
ii 
服务 器 
MTDY-952TVISSQLSERYER2008 末 已 BE 用 吧 ) 
连接 
NIDN-952T\Adninistr ator 
寺 查看 连接 属性 
Fr 
就 结 
Cw |] ww | 
4 
图 16.5 新 建 作 业 


16.2.2 ”定义 一 个 作业 步骤 

完成 了 作业 的 创建 后 , 作业 还 不 能 帮助 用 户 做 什么 。 这 就 好 像 是 老师 只 留 了 语文 作业 ， 
但 是 没 具体 说 明 是 什么 作业 内 容 一 样 ， 你 怎么 完成 语文 作业 呢 ? 因此 ， 还 要 对 作业 中 具体 
要 完成 的 内 容 加 以 说 明 。 在 SQL Server 中 ， 作 业 中 的 内 容 是 通过 作业 步骤 来 添加 的 。 添 加 
作业 步骤 需要 通过 以 下 几 个 步骤 完成 。 

1. 打开 显示 作业 步骤 的 界面 

在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ” 创建 一 个 新 作业 或 右 击 一 个 现 有 


作业 ， 在 弹出 的 右键 菜单 中 选择 “属性 ”选项 。 在 “作业 属性 ”界面 中 ， 单 击 “步骤 ” 选 
项 ， 弹 出 如 图 16.6 所 示 的 界面 。 


2. 打开 新 建 作业 步骤 的 界面 
在 图 16.6 所 示 界 面 中 ， 单 击 “ 新 建 ” 按 钮 ， 弹 出 界面 如 图 16.7 所 示 。 
3. 添加 作业 步骤 信息 


在 创建 作业 步骤 之 前 ， 先 创建 一 个 在 本 章 使 用 的 数据 库 chapter16。 然 后 ， 在 作业 步骤 
中 创建 在 chapter16 中 的 用 户 信息 表 (userinfo)。 为 了 简单 起 见 ， 用 户 信息 表 中 只 包含 用 户 
名 和 密码 两 个 字段 。 添 加 后 的 作业 步骤 信息 如 图 16.8 所 示 。 
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局 新 建 作业 步 要 


16.7 “新建 作业 步骤 
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殴 新 建 作业 步 收 EE -lolxl 
过 办 页 | - DB#w 
EE 
EE 步 要 名 称 G): 
5 | 
类 型 中 : 
[reansset-sar a 7-saL) | 


运行 身份 入: 


蕊 范 贴 EE) 
服务 器 
HT-952TWISSQLSERYEF2008 
Le | 


连接 
MTDW-9S2T\Adninistrator 


圣 本 看 连接 属性 


a 
. 


就 绪 


图 16.8 添加 作业 步骤 信息 


外 说 明 : 如 果 需 要 对 作业 步骤 进行 其 他 设置 ， 可 以 在 此 界面 中 单 击 “ 高 级 ”选项 ， 即 可 设 
置 作 业 步 骤 的 其 他 信息 。 


单 击 “ 确 定 ” 按 钮 ， 出 现 图 16.9 所 示 的 界面 。 

4. 完成 作业 步骤 的 创建 

在 图 16.9 所 示 界面 中 ,如果 需要 继续 添加 作业 步骤 , 可 以 单 击 “ 新 建 ” 按 钮 继续 添加 。 
如 果 不 需 要 再 添加 作业 步骤 信息 了 ， 直 接 单 击 “确定 ”按钮 ， 即 可 完成 作业 步骤 的 创建 。 
16.2.3 创建 一 个 作业 执行 计划 

作业 创建 好 后 ， 如 何 执行 呢 ? 在 SQL Server 中 ， 提 供 了 一 个 比较 简单 的 方式 ， 那 就 
是 制定 作业 执行 计划 。 通 过 制定 作业 执行 计划 能 够 使 作业 按照 计划 的 时 间 执 行 ， 比 如 : 
每 周一 执行 、 每 周 执 行 儿 次 等 等 。 通 过 企业 管理 器 创建 作业 执行 计划 ， 通 常 需要 如 下 4 
个 步骤 。 

1. 打开 显示 作业 执行 计划 的 界面 

在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ”节点 ， 创 建 一 个 新 作业 或 右 击 一 
个 现 有 作业 , 在 弹出 的 右键 菜单 中 选择 “属性 ” 选项， 然后 在 “作业 属性 ”界面 中 选择 “ 计 
划 ” 选 项 ， 如 图 16.10 所 示 。 
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Ja 时 jx 时 | 


jabl-1 Transact-S... 转 到 下 一 步 退出 报告 失 - 


ebbivivts| 
四 草 噶 国 贞 对 
EF EE 


服务 器 
TI-952TWISSQLSERYEH2008 
连接 
IDF-9527\Adninistrator 


圭 查 下 连接 性 


就 绪 


图 16.9 添加 步骤 后 的 效果 


= 车 
-Dy 
计划 列表 
[ET 已 8 用 [说明 让 


图 16.10 ”作业 计划 列表 界面 
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2. 打开 新 建 作业 执行 计划 的 界面 


在 图 16.10 所 示 界 面 中 ， 可 以 查看 到 当前 作业 中 现 有 的 作业 计划 。 如 果 要 新 建 作业 计 
划 ， 则 单 击 “ 新 建 ” 按 钮 ， 弹 出 图 16.11 所 示 的 界面 。 


图 16.11 新 建 作业 执行 计划 界面 


3. 填写 作业 执行 计划 


在 图 16.11 所 示 界 面 中 ， 填 入 相应 的 作业 执行 计划 即 可 。 假 设 要 计划 每 周一 上 午 8 点 
执行 一 次 作业 ， 则 填 入 后 的 效果 如 图 16.12 所 示 。 


图 16.12 ” 填 入 执行 计划 的 效果 


人 
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从 图 16.12 所 示 界 面 中 ， 可 以 在 最 下 面 的 说 明 部 分 看 到 制定 的 计划 是 “在 每 周 星 期 一 
的 8:00 执行 。 将 从 2013-3-9 开始 使 用 计划 ”。 


4. 保存 作业 执行 计划 


在 图 16.12 所 示 界 面 中 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 作业 计划 的 创建 。 如 果 需 要 使 
所 创建 的 计划 立即 生效 ， 那 么 ， 就 要 将 “已 启用 ”前 面 的 复 选 框 选中 。 


16.2.4 ”查看 和 管理 作业 


在 作业 创建 完成 后 ， 经 常会 需要 查看 、 修 改 以 及 删除 作业 的 内 容 ， 这 些 对 作业 的 操作 
在 企业 管理 器 下 都 是 非常 容易 操作 的 。 此 外 ， 读 者 又 有 了 创建 作业 的 基础 ， 对 于 作业 的 其 
他 操作 就 更 容易 掌握 了 。 下 面 就 分 别 讲解 查看 作业 和 管理 作业 。 


1. 查看 作业 


查看 作业 的 操作 是 非常 简单 的 ， 在 前 面 创建 作业 步骤 或 者 创建 作业 计划 时 都 会 查看 到 
作业 的 内 容 。 实 际 上 ， 作 业 就 是 通过 作业 属性 界面 查看 到 的 ， 这 里 就 不 再 多 说 了 。 这 里 的 
查看 作业 ， 主 要 是 给 读者 讲解 如 何 查看 作业 的 活动 。 在 企业 管理 器 中 ， 查 看 作业 活动 分 为 
如 下 两 个 步骤 。 

(1) 打开 作业 活动 监视 器 

在 “对 象 资 源 管 理 器 ”中 ， 展 开 “SQL Server 代理 ” 右 击 “ 作 业 活 动 监视 器 ”节点 ， 
在 弹出 的 右键 菜单 中 选择 “查看 作业 活动 ”选项 ， 弹 出 图 16.13 所 示 的 界面 。 


红 作 业 活 动 监视 器 - WIDW-9527\MSSQLSERYER2008 吕 革 
ET 


:== 
上 次 刷新 
2013-3-9 8:47:55 
下 次 刷新 
手动 
加 查看 剧 新 设置 


名 地 器， 无 
了 查看 各 过 设 轩 


ED 


服务 器 :WIDN-9527\MSSQLSERYER2008 
连接 ITDY-952TVAdainistrator 


对 ”查看 连接 属性 


图 16.13 ”作业 活动 监视 器 
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(2) 查看 作业 信息 


在 图 16.13 所 示 界 面 中 ， 可 以 右 击 任意 作业 ， 在 弹出 的 右键 菜单 中 选择 “属性 ”选项 ， 
就 可 以 查看 作业 的 信息 ; 选择 “作业 开始 步骤 ”选项 ， 则 执行 该 作业 ; 选择 “禁用 作业 ” 
选项 ， 则 该 作业 被 禁用 ; 选择 “启用 作业 ”选项 ， 则 该 作业 被 启用 :; 选择 “删除 作业 ” 选 
项 ， 则 该 作业 被 删除 ;选择 “查看 历史 信息 记录 ”选项 ， 则 显示 该 作业 执行 的 日 志 信息 。 
2. 管理 作业 


对 于 作业 的 管理 ， 主 要 包括 对 作业 的 修改 和 删除 操作 。 修 改作 业 与 查看 作业 基本 都 是 
一 样 的 ， 都 是 在 作业 的 属性 界面 中 完成 的 。 在 修改 作业 时 可 以 修改 作业 的 步骤 、 计 划 等 信 
息 。 但 是 ， 修 改 后 的 作业 一 定 要 记得 保存 哦 ! 相信 读者 通过 前 面 的 学 习 一 定 能 够 独立 完 
修改 作业 的 操作 ， 就 不 再 费 述 了 。 这 里 ， 主 要 讲解 如 何 删除 作业 。 实 际 上 ， 在 企业 管理 器 
中 删除 作业 是 很 简单 的 ， 只 需要 如 下 步骤 即 可 搞定 。 

在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ”节点 。 右 击 一 个 作业 名 称 ， 在 弹 
出 的 右键 菜单 中 选择 “删除 ”选项 ， 弹 出 如 图 16.14 所 示 的 界面 。 


=|Ox| 
Su -Dy 
Ee 
要 山 除 的 对 象 Q) 
= 
服务 器 
CE 
连接 
和 查 看 连接 屋 竹 
EE 
就 绪 


图 16.14 删除 提示 界面 
在 图 16.14 所 示 界 面 中 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 删除 该 作业 。 


16.3 维护 计划 


维护 计划 可 以 说 是 数据 库 管 理 员 的 好 帮手 ， 使 用 维护 计划 可 以 实现 一 些 自 动 的 维护 工 
作 。 通 过 维护 计划 可 以 完成 数据 的 备份 、 重 新 生成 索引 、 执 行 作 业 等 操作 。 虽 然 维护 计划 


ss 
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的 功能 强大 ， 但 是 在 企业 管理 器 中 创建 维护 计划 却 是 很 容易 的 。 
16.3.1 什么 是 维护 计划 

维护 计划 与 之 前 创建 的 作业 计划 有 些 类 似 ， 都 是 通过 制定 计划 来 自动 完成 一 些 特定 功 
能 。 那 么 ， 维 护 计 划 究 竟 能 帮助 用 户 完成 哪些 功能 呢 ? 下 面 就 列 出 维护 计划 最 常 应 用 的 几 
个 方面 。 

(1) 用 于 自动 运行 SQL Server 作业 。 

(2) 用 于 定期 备份 数据 库 。 

(3) 用 于 检测 数据 库 完 整 性 。 

(4) 用 于 更 新 统计 数据 。 

(5) 用 于 重新 组 织 和 生成 索引 。 


16.3.2 ”使 用 向 导 创 建 维 护 计划 步骤 

维护 计划 向 导 就 像 安装 软件 时 的 向 导 一 样 ， 通 过 向 导 的 指示 一 步 一 步 地 设置 就 可 以 完 
成 一 个 维护 计划 的 创建 了 。 在 企业 管理 器 中 ， 通 过 向 导 创建 维护 计划 需要 通过 如 下 几 个 步 

1. 打开 SQL Server 维护 计划 向 导 

在 “对 象 资源 管理 器 ”中 ， 展 开 “ 管 理 ” 节 点 ， 右 击 “ 维 护 计 划 ” 节 点 ， 在 弹出 的 右 
键 菜单 中 选择 “维护 计划 向 导 ” 选 项 ， 弹 出 界面 如 图 16.15 所 示 。 


硬 准 护 计划 向 导 =l9lx 
SQL Server 维护 计划 向 导 
i 
此 机 可 入 | ok Suryer 代理 定 随 去 行 的 扒 护 计划 。 合 用 此 向 号 还 可 
A i 0 
Se 
更 同时 和 时 于 三 st Sorvar Mansensnt stodis 中 加 入 的 维 护 计划 , 的 
| NP 更 CD :下 和 生 生 肖 生 于 和 
全 
三 不 两 显示 此 起 好 页 D)- 
um | ts9 | 3 | ww 
者 


图 16.15 维护 计划 向 导 界 面 


2. 选择 计划 属性 


在 图 16.15 所 示 界 面 中 ， 单 击 “ 下 一 步 ” 按 钮 ， 出 现 图 16.16 所 示 的 界面 。 在 此 界面 
中 ， 可 以 为 计划 填 入 名 称 、 说 明 ， 并 选择 该 计划 是 每 项 任务 的 单独 计划 还 是 要 整个 计划 统 
筹 安排 或 无 计划 。 
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= 上 到 


TT SS 


名 称 胃 Pintenanerlon 
说 明 四 )- 


江 每 项 任务 单独 计划 


计划 : 


[ES A 
帮助 0D < 上- 步 加 RE cl) 
图 16.16 选择 计划 属性 界面 


3. 选择 维护 任务 
在 图 16.16 所 示 界 面 中 ， 单 击 “ 下 一 步 ”按钮 ， 即 可 出 现 图 16.17 所 示 的 选择 维护 任 
务 界面 。 


可 
园 择 六 作 各 5gpevte 


口 备份 数据 库 ( 萱 异 ) 
口 备份 数据 库 事务 日 志 ) 
口 “ 清 除 维护 ”任务 


芭 “检查 数据 库 完整 性 ”任务 用 于 检查 数据 库 中 的 数据 和 索引 页 内 部 是 否 一 致 。 


才 助 0 《上 一 步 叹 2 取消 史 
0 
图 16.17 选择 维护 任务 


在 图 16.17 所 示 界 面 中 ， 列 出 了 所 有 维护 计划 可 以 执行 的 任务 。 在 此 ， 可 以 选择 一 个 
或 多 个 执行 任务 。 这 里 ， 选 择 “ 执 行 SQL Server 代理 作业 ”选项 。 


4. 选择 维护 任务 顺序 


在 图 16.17 所 示 界 面 中 ， 单 击 “ 下 一 步 ”按钮 ， 即 可 转 到 图 16.18 所 示 的 选择 维护 任 
务 顺序 界面 。 
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[eribaps 一 本 
选择 维护 任务 顺序 
应 该 按 什 么 顺序 执行 这 些 任 务 ? 全 
选择 执行 任务 的 顾 序 GE) 
上 大 加 TB 


也 如 Eo SQL Server 代理 作业 ”任务 ， 悠 可 以 选择 SQL Server 代理 作业 ， 梅 其 作为 维护 计 X 的 一 
部 分 运行 。 


Ew Eee 有 
图 16.18 选择 维护 任务 顺序 


在 图 16.18 所 示 界 面 中 ， 如 果 选 择 了 多 个 任务 就 可 以 为 这 些 任务 排列 执行 顺序 。 这 里 
只 选择 了 一 个 任务 ， 因 此 不 用 对 任务 排序 。 


5. 选择 要 执行 的 具体 任务 


在 图 16.18 所 示 界 面 中 ， 单 击 “ 下 一 步 ”按钮 ， 出 现 选择 具体 任务 界面 ， 如 图 16.19 
所 示 。 由 于 在 前 面 选 择 的 是 SQL Server 作业 任务 ， 因 此 在 该 界面 中 列 出 的 是 系统 中 所 有 的 
作业 信息 。 


Trius By 
定 务 , 热 每 ， SoL Server 代理 作业 ”任务 


可 用 的 SQL Server 代理 作业 (A): 


计划 


[5 ERP] 
wm | 《< 上 - 步 四 2 取消 网 
| 


图 16.19 选择 要 使 用 的 SQL Server 代理 作业 
这 里 ， 选 中 名 称 为 jobl 的 作业 。 
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6. 选择 报告 选项 
在 图 16.19 所 示 界 面 中 ， 选 择 好 作业 后 ， 单 击 “ 下 一 步 ” 按 钮 ， 出 现 选 择 报告 输出 位 

置 的 界面 ， 如 图 16.20 所 示 。 
=I9lx| 


邮 维护 计划 育 导 


选择 报告 选项 
先 失 选项 对 维护 计划 并 作 报告 进行 保存 或 分 发 和 


Erroew Files\icrosoft SQL Server\nssario. nssars 上 


厂 以 电子 邮件 形式 发 送 报 省 中) 
a 二 


笨 助 四 《上 - 步 @) 5 职 滑 » 
图 16.20 选择 报告 选项 界面 
在 此 ， 指 定 报告 输出 的 位 置 或 者 以 电子 邮件 的 方式 发 送 给 报告 人 。 这 里 ， 选 择 将 报告 
输出 到 文件 中 。 


7. 完成 向 导 
在 图 16.20 所 示 界 面 中 ， 单 击 “ 下 一 步 ”按钮 ， 即 可 完成 向 导 界面 ， 如 图 16.21 所 示 。 


EE:| 


完 记 潜 自 时 pgm, wa "mv 。 
\ 


单 击 “ 完 成 ”以 执行 下 列 换 作 : 


EE 
田 创建 维护 计划 “aintenancePlan” 
由 -定义 “执行 SQL Server 代理 作业 ”任务 
困 - 所 选 报 省 选项 


图 16.21 完成 向 导 界 面 
在 此 界面 中 ， 确 认 维护 计划 向 导 的 内 容 ， 然 后 单 击 “ 完 成 ”按钮 ， 出 现 执 行 计 划 的 界 


面 ， 如 图 16.22 所 示 。 
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邮 维护 计划 向 导 本 =|9|z 
Vv 


维护 计划 向 导 进 度 
单 击 “ 停 止 ”以 中 断 操作 。 


Om 5 总 计 0 错误 


详细 信息 四 )- 


加 创 陵 维 护 计划 “MsintenaneePlan” 
加 将 任务 添加 加 维护 计划 

加 添加 计划 选项 

加 添加 报告 选项 

加 保存 维护 计划 “HaintenanceFlan” 


图 16.22 ”向导 执行 界面 
如 果 能 够 出 现 图 16.22 所 示 的 成 功 结果 ， 就 意味 着 完成 了 维护 计划 的 创建 操作 。 


外 说 明 : 创建 成 功 的 向 导 会 直接 出 现在 维护 计划 的 节点 下 。 如 果 需 要 修改 或 删除 该 向 导 ， 
则 直接 右 击 该 向 导 ， 在 弹出 的 右键 菜单 中 ， 选 择 相应 的 操作 即 可 。 但 是 ， 有 一 点 
需要 注意 的 是 ， 维 护 计划 不 仅 会 出 现在 维护 计划 的 节点 下 ， 也 会 出 现在 作业 的 节 
点 下 。 建 议 读者 不 要 直接 操作 作业 节点 下 的 维护 计划 ， 以 免 造 成 操作 错误 。 


16.4 警 报 


警报 通常 是 在 违反 了 一 定 的 规则 后 出 现 的 一 种 通知 行为 。 现 在 听 到 比较 多 的 就 是 消防 
警报 ， 在 单位 或 者 是 宾馆 一 般 都 会 安装 消防 警报 ， 用 来 监控 消防 的 一 些 安全 隐患 ， 并 不 是 
说 触发 了 消防 警报 就 一 定 会 有 火灾 。 比 如 : 有 些 禁烟 单位 ， 当 有 人 抽烟 了 消防 警报 就 会 响 。 
在 使 用 数据 库 时 也 是 一 样 的 ， 当 预先 设 定好 的 错误 发 生 时 ， 就 会 发 出 警告 告知 用 户 。 
16.4.1 创建 警报 

在 数据 库 中 合理 地 使 用 警报 能 够 帮助 数据 库 管理 员 更 好 地 管理 数据 库 ， 并 提高 数据 库 
的 安全 性 。 通 过 企业 管理 器 创建 警报 ， 通 常 需要 如 下 几 个 步骤 完成 。 


1. 打开 新 建 警报 界面 


在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ”节点 ， 右 击 “ 和 警报 ”节点 ， 在 弹 
出 的 右键 菜单 中 选择 “新 建 警 报 ” 选 项 ， 如 图 16.23 所 示 。 


2. 输入 警报 信息 
在 图 16.23 所 示 界 面 中 ， 输 入 警报 的 名 称 、 类 型 等 信息 。 类 型 包括 “SQL Server 事件 
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全 新 建 苔 报 


=Iolx| 
这 办 页 -四 玫 助 
EE 
字 中 应 
江汉 名 称 四: 末 已 用 的) 
关 型 中; FE server 名 必要 报 司 
事件 警报 定义 
数据 库 名 称 四) [2 司 
将 根据 以 下 条 件 触 发 警报 ; 
个 错误 号 四; 
人 严重 性 : oo1 - 杂项 系统 信息 了 
厂 当 清 息 包 全 以 下 内 容 时 鹏 发 警报 人 ): 
请 息 正 文 全 ) 
服务 器 
WIDN~9527\MSSQLSERVER2008 
这 按 
WIDW-9527\Adninistrator 
寺 查看 连接 屋 考 
训 弹 


图 16.23 ”新 建 警报 界面 


警报 “SQL Server 性 能 条 件 警 报 ” 和 “WMI 事件 警报 ” 输入 相应 的 警报 信息 后 ， 还 可 
以 通过 图 16.23 左 侧 的 “响应 ”和 “选项 ”内 容 的 设置 完成 警报 信息 的 添加 。 


3. 确认 警报 信息 
添加 好 警报 信息 后 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 警报 的 创建 。 
全 说 明 : 警报 创建 完成 后 ， 会 出 现在 警报 节点 下 。 如 果 要 查看 该 警报 ， 可 以 通过 右 击 该 警 


报 ， 在 弹出 的 右键 菜单 中 选择 “属性 ”选项 ， 即 可 查看 警报 。 同 时， 也 可 以 通过 
右键 菜单 选择 “启用 ”或 “禁用 ”该 警报 。 


16.4.2 ”删除 警报 


在 不 同 的 数据 库 应 用 中 ， 警 报 的 设置 也 是 不 相同 的 。 因 此 ， 当 某 些 警 报 不 再 需要 时 ， 
就 可 以 将 其 删除 了 。 在 SQL Server 中 ， 删 除 警报 是 非常 容易 的 。 通 过 企业 管理 器 只 需要 下 
面 1 个 步骤 就 可 以 完成 了 。 

在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ” |“ 警报” 节点， 右 击 一 个 警报 ， 
在 弹出 的 右键 菜单 中 选择 “删除 ”选项 ， 弹 出 如 图 16.24 所 示 的 界面 。 

在 图 16.24 所 示 界 面 中 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 警报 的 删除 操作 。 
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划 除 对 象 
迁 舞 页 
pe 


=/9lx| 


服务 器 
WTIW-9527\MSSQLSERYER2008 


连接 
WIDN-9527\Adnini strator 


对 查看 连接 属性 


图 16.24 删除 警报 提示 界面 


16.5 操 作 员 


操作 员 实 际 上 就 是 SQL Server 数据 库 中 设 定好 的 信息 通知 对 象 。 当 系统 出 现 警 报时 ， 
可 以 直接 通知 操作 员 ， 或 者 是 当 执行 任务 成 功 后 都 可 以 通知 操作 员 。 通 知 操作 员 的 方式 通 
常 是 发 送 电子 邮件 或 者 是 通过 Windows 系统 的 服务 发 送 网 络 信息 。 


16.5.1 创建 操作 员 


创建 操作 员 是 使 用 操作 员 的 第 一 步 。 在 企业 管理 器 中 ， 创 建 操作 员 是 很 容易 的 。 创 建 
操作 员 通常 需要 如 下 3 个 步骤 完成 。 


1. 打开 新 建 操作 员 界 面 


在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ”节点 ， 右 击 “操作 员 ” 节 点 ， 在 
弹出 的 右键 菜单 中 选择 “新 建 操作 员 ” 选 项 ， 弹 出 界面 如 图 16.25 所 示 。 

2. 填 入 操作 员 信 息 

在 图 16.25 所 示 界 面 中 ， 填 入 操作 员 的 姓名 以 及 “通知 ”选项 中 的 任意 信息 。 并 且 ， 


可 以 “在 寻 呼 值班 计划 ”中 选择 操作 员 的 工作 时 间 。 这 里 ， 为 了 方便 后 面 使 用 操作 员 ， 将 
操作 员 的 姓名 填 入 “users1”， 并 填 入 邮箱 名 称 “userl@sina.com”。 
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MD) 工作 日 hi 时 间 工作 日 结 束 时 间 

EE [Er | | 

厂 星 类 入 WD 8:00:00 忆 [soo'o 村 
Ee 厂 BE Er 司 [Emm 村 


服务 器 
MTDW-9527WISSQLSERYER2008 


连接 
人 TDW-9527\VAdninistrator 


对 查看 连接 属性 


图 16.25 新 建 操作 员 界 面 


3. 确认 操作 员 信 息 
填 入 操作 员 信 息 后 ， 单 击 图 16.25 所 示 界 面 的 “确定 ”按钮 ， 即 可 完成 操作 员 的 创建 。 
全 说 明 : 操作 员 创 建 完成 后 ， 就 会 出 现在 操作 员 的 节点 下 。 如 果 想 管理 操作 员 ， 可 以 右 击 
该 操作 员 ， 并 在 弹出 的 右键 菜单 中 选择 相应 的 选项 对 其 进行 管理 。 
16.5.2 ”使 用 操作 员 


在 上 一 小 节 中 已 经 学 会 了 如 何 创 建 操作 员 ， 那 么 ， 操 作 员 究 竟 如 何 使 用 呢 ? 下 面 就 以 
在 警报 中 使 用 操作 员 为 例 讲解 操作 员 的 应 用 。 在 企业 管理 器 中 ， 使 用 操作 员 通 常 需要 如 下 
两 个 步骤 即 可 完成 。 


1. 打开 操作 员 属 性 界面 


在 “对 象 资源 管理 器 ”中 ， 展 开 “SQL Server 代理 ”|“ 操 作 员 ”节点 ， 右 击 userl 操 
作 员 ， 在 弹出 的 右键 菜单 中 选择 “属性 ”选项 ， 弹 出 界面 如 图 16.26 所 示 。 


2. 选择 通知 操作 员 的 方式 
在 图 16.26 所 示 界 面 中 ， 选 择 “ 通 知 ” 选 项 ， 如 图 16.27 所 示 。 
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服务 器 : 
WIDN-9527\NSSQLSERVER2008 
连接 
DY-9527VAdninistrater 


16.26 ”userl 属性 界面 


同时 wwssusmmeoo 


连接 : 
ID-9527\Adninistrator 


图 16.27 选择 通知 操作 员 的 方式 
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这 里 ， 选 择 “ 警 报 ” 


， 并 在 复 选 框 中 选择 通知 的 方式 “电子 邮件 ”。 单 击 “ 确 定 ” 按 钮 ， 


即 可 完成 操作 员 的 通知 设置 。 


16.6 本 章 小 结 


本 章 详细 介绍 了 SQL Server 中 自动 化 管理 的 基本 对 象 ， 包 括 作业 、 维 护 计 划 、 和 警报 以 


及 操作 员 等 。 其 中 ， 着 导 


讲解 了 作业 的 创建 、 作 业 步 骤 的 创建 以 及 计划 创建 等 。 通 过 本 章 


的 学 习 ， 读 者 可 以 有 选择 地 借助 这 些 自 动 化 管理 工具 来 管理 数据 库 。 特 别 是 ， 读 者 合理 地 
使 用 警报 能 够 避免 对 数据 库 操 作 中 的 一 些 错误 。 


16.7 本 章 练习 


一 、 填 空 题 

1. 使 用 SQL Server 代理 时 ， 必 须要 启动 的 服务 是 
2. 如 果 在 数据 库 操作 中 触发 警报 可 以 通知 a 

3. 操作 员 的 通知 方式 有 种 。 

二 、 选 择 题 

1. 下 列 对 象 中 哪个 不 是 SQL Server 代理 中 的 内 容 


A. 操作 员 


B. 警报 C. 作业 D. 维护 计划 


2. 操作 员 可 以 通过 下 列 哪个 对 象 进行 通知 ? 


A. SQL Server 代理 B. 警报 

C. 计划 D. 以 上 都 不 是 
3. 一 个 作业 通常 都 包括 哪些 内 容 ? 

A. 步骤 B. 计划 C. 警报 D. 以 上 都 是 
三 、 问 答题 


1. 什么 是 维护 计划 ? 都 可 以 维护 哪些 内 容 ? 


2. 如 何 创建 作业 ? 
3. 如 何 创建 警报 ? 


四 、 操 作 题 


根据 本 章 所 学 的 内 容 ， 完 成 如 下 的 操作 。 

(1) 创建 一 个 名 为 jobl 的 作业 ， 完 成 在 数据 库 chapter16 中 创建 任意 数据 表 的 操作 。 
(2) 设置 该 作业 执行 的 时 间 为 每 周 五 的 晚上 8 点 。 

(3) 创建 一 个 维护 计划 执行 该 作业 。 
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如 果 仅 把 数据 存放 在 数据 库 中 就 置之不理 了 , 那么 , 数据 库 中 的 数据 也 就 毫 无 意义 了 。 
换 名 话说， 在 数据 库 中 存放 数据 ， 是 为 了 用 户 更 好 地 查询 和 使 用 数据 。 因 此 ， 要 借助 其 他 
编程 语言 构建 操作 页 面 来 操作 数据 库 。SQL Server 最 亲密 的 搭档 就 是 出 自 于 同一 个 公司 的 
Visual Studio 平台 。 本 章 中 将 使 用 C# 语 言 在 该 平台 下 完成 与 SQL Server 的 连接 操作 。 

本 章 的 主要 知识 点 如 下 : 

口 认识 ADO.NET 

口 使 用 ADO.NET 连接 SQL Server 数据 库 

口 认识 Windows 窗 体 

口 使 用 Windows 窗 体 程序 完成 对 数据 的 添加 和 管理 的 操作 


17.1 ADO.NET 介绍 


说 到 使 用 C# 语 言 连接 SQL Server 数据 库 ， 不 得 不 提 的 就 是 其 ADO.NET 组 件 。 通 过 
ADO.NET 组 件 可 以 很 方便 地 连接 SQL Server 数据 库 ， 并 对 其 进行 数据 的 添加 、 修 改 、 删 
除 以 及 查询 的 操作 ， 此 外 还 可 以 通过 C# 语 言 创建 数据 库 、 数 据 表 、 视 图 以 及 存储 过 程 等 对 
象 。 在 本 节 中 主要 讲解 ADO.NET 组 件 中 五 大 对 象 的 作用 以 及 使 用 方法 。 


17.1.1 认识 ADO.NET 


在 微软 推出 的 .NET 平台 下 ， 有 很 多 语言 都 可 以 作为 其 应 用 的 开发 语言 ， 比 如 : VB、 
C# 等 。 在 本 章 中 选择 使 用 C# 语 言 , 主要 是 考虑 C# 语 言 是 目前 在 .NET 平台 上 开发 的 主流 语 
言 , 也 是 很 多 公司 都 选择 的 一 种 语言 。C# 语 言 是 一 种 面向 对 象 的 语言 , 如 果 读 者 学 习 过 Java 
语言 再 学 习 它 就 会 变 得 很 容易 了 。 在 .NET 平台 上 , 无 论 选用 何 种 语言 来 连接 数据 库 ， 都 需 
要 使 用 ADO.NET 组 件 。 因 此 ， 了 解 ADO.NET 的 组 成 就 至 关 重 要 了 。 通 常 在 连接 数据 库 
时 需要 使 用 ADO.NET 中 的 Connection、Command、DataAdapter、DataSet 和 DataReader 
等 35 类。 下面 就 分 别 来 介绍 这 五 个 类 的 具体 作用 。 


1. Connection 类 


Connection 类 被 称 为 数据 库 连接 类 ， 因 此 ， 只 要 做 数据 库 连 接 都 少不了 它 。 它 的 主要 
功能 就 是 连接 数据 库 、 打 开 数 据 库 连 接 以 及 关闭 数据 库 连接 。 在 每 个 与 数据 库 相 关 的 操作 
中 ， 第 一 步 都 是 要 先 连接 数据 库 ， 这 就 好 像 是 找到 数据 库 的 连接 大 门 ， 第 二 步 就 是 打开 数 
据 库 连接 ， 就 好 像 是 打开 数据 库 的 这 扇 门 : 最 后 一 步 就 是 在 对 数据 库 操作 完成 后 ， 关 闭 数 
据 库 连接 ， 就 好 像 是 从 家 出 来 要 关门 一 样 。 使 用 Connection 类 不 仅 可 以 连接 SQL Server 
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数据 库 ， 也 可 以 连接 其 他 数据 库 ， 比 如 : Oracle、MySQL 和 Access 等 。 但 是 ， 连 接 每 种 
数据 库 需 要 引用 的 命名 空间 不 同 ， 这 就 好 像 是 邮局 ， 能 够 邮寄 市 内 、 省 内 、 国 内 以 及 国际 
各 地 的 信件 ， 但 是 所 填写 的 邮编 却 不 同 。 究 竟 使 用 SQL Server 数据 库 需 要 引用 什么 样 的 命 
名 空间 ， 答 案 将 会 在 下 一 小 节 揭晓 。 


2. Command 类 


Command 类 被 称 为 数据 库 命令 类 。 换 句 话说 , 它 就 是 发 出 对 数据 库 操作 命令 的 。 根 据 
Command 类 在 执行 不 同 的 数据 库 操作 命令 时 执行 方法 的 不 同 , 可 以 简单 将 操作 命令 分 为 查 
询 命令 和 非 查 询 命令 两 种 。 查 询 命 令 就 是 通常 说 的 SELECT 查询 操作 ， 而 非 查询 命令 通常 
就 是 指 执 行 INSERT、UPDATE 和 DELETE 的 操作 。 但 是 ， 该 命令 一 定 是 在 数据 库 打 开 之 
后 才能 够 进行 的 。 同 样 ， 既 然 Connection 类 能 够 连接 多 种 数据 库 ， 那 么 ，Command 类 也 
就 能 够 操作 多 种 数据 库 了 。 


3. DataAdapter 类 


DataAdapter 类 被 称 为 数据 适配器 类 ， 与 Command 类 的 功能 很 相似 ， 都 是 对 数据 库 进 
行 操作 的 。 但 是 ， 它 主要 是 应 用 在 数据 集 类 DataSet 中 。 它 可 以 形象 地 说 成 是 数据 库 与 数 
据 集 的 桥梁 ， 桥 梁 上 传输 的 就 是 数据 ， 数 据 从 数据 库 传 出 到 数据 集中 ,数据 集中 数据 变化 ， 
又 可 以 将 数据 传 给 数据 库 。 


4. DataReader 类 


DataReader 类 被 称 为 数据 读 取 类 , 通常 是 用 来 存放 查询 结果 的 。 该 类 经 常 与 Command 
类 连用 ， 用 于 查询 结果 的 存储 。 从 该 类 中 读 取 查 询 结 果 只 能 按照 顺序 来 读 取 ， 并 且 每 次 只 
能 读 取 一 条 数据 。 


5. DataSet 类 


DataSet 类 与 DataReader 类 一 样 ， 是 用 来 存放 数据 的 ， 但 是 它们 有 很 多 不 同 点 。 最 重 
要 的 一 点 就 是 使 用 DataSet 存放 数据 时 ， 当 数据 库 断 开 连 接 时 ， 也 依然 可 以 使 用 DataSet 
中 的 数据 。 其 次 ， 就 是 在 DataSet 中 存放 的 数据 是 可 以 任意 读 取 而 不 必 按照 顺序 读 取 。 因 
此 ， 在 大 多 数 的 程序 中 使 用 数据 集 来 存储 数据 是 比较 方便 的 ， 也 是 最 好 的 选择 。 
17.1.2 ”使 用 Connection 连接 SQL Server 数据 库 


在 上 一 小 节 中 , 已 经 学 习 了 ADO.NET 中 的 五 大 类 。 那 么 ,在 本 小 节 中 就 使 用 Connection 
类 来 演示 如 何 连接 SQL Server 数据 库 。 请 记 住 ， 操 作 分 为 如 下 5 个 步骤 。 


1. 引用 命名 空间 


在 C# 语 言 中 , 命名 空间 实际 上 可 以 理解 为 存放 类 的 文件 夹 。 引用 命名 空间 使 用 的 语句 
如 下 : 


using 命名 空间 名 称 ; 
这 里 ， 命 名 空间 名 称 就 是 类 所 在 文件 夹 的 路 径 ， 只 是 将 路 径 中 的 “\” 换 成 “.”。 并 且 
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要 求 该 文件 与 当前 的 项 目 在 同一 文件 夹 下 。 例 如 : 类 所 在 的 文件 夹 的 路 径 是 Ca\b， 则 写 
成 命名 空间 的 形式 就 是 using ab， 当 前 项 目 也 同 在 C 盘 下 。 
如 果 要 使 用 SQL Server 数据 库 ， 则 通常 都 会 引用 的 命名 空间 如 下 : 


using System.Data.SqlClient; 


全 注意 : 在 C# 语 言 中 ， 是 严格 区 分 大 小 写 的 ! 


2. 编写 数据 库 连 接 字符 串 


所 谓 数据 库 连 接 字 符 串 ， 就 是 写 着 连接 数据 库 的 名 称 以 及 数据 库 服务 名 的 字符 串 。 具 
体 的 写法 如 下 : 


Server=server name; database=database name; Integrated Security=True 


其 中 : 
口 server_ name: 服务 器 名 , 如 果 是 本 地 数据 库 则 可 以 使 用 local 或 者 是 用 “.” 来 表示 。 
口 database_ name: 要 连接 的 数据 库 名 。 
口 Integrated Security=True: 代表 当前 数据 库 使 用 的 是 Windows 方式 登录 SQL Server 
全 注意 : 如 果 数 据 库 使 用 的 是 SQL Server 的 登录 方式 时 ， 还 要 在 连接 字符 串 上 加 上 用 户 名 
和 密码 。 


3. 创建 数据 库 连接 对 象 

有 了 上 面 的 连接 字符 串 ， 创 建 数据 库 连 接 对 象 就 不 是 什么 难事 了 。 创 建 数 据 库 连接 对 
象 要 使 用 Connection 类 来 完成 ， 如 果 引 用 了 System.Data.SqlClient 这 个 命名 空间 ， 那 么 ， 
Connection 类 就 要 写成 SqlConnection。 有 具体 的 创建 语句 如 下 : 

SqlConnection connection_name =new SqlConnection (connection str); 

其 中 : 

口 connection name: 连接 对 象 名 称 。 不 能 以 数字 开头 ， 与 SQL Server 中 定义 对 象 的 

标识 符 的 规则 是 一 样 的 。 
口 connection_str 连接 字符 串 。 这 部 分 内 容 就 是 第 2 步 中 编写 的 连接 字符 串 。 
使 用 上 面 定义 的 连接 字符 串 创建 连接 对 象 的 语句 如 下 : 


SqlConnection conn =new SqlConnection (Server=.\MSSQLSERVER2008; 
database=chapter17; Integrated Security=True); 


这 里 ， 笔 者 的 数据 库 服务 名 是 \MSSQLSERVER2008， 要 连接 的 数据 库 是 chapter17。 
创建 的 连接 对 象 名 是 conn。 记 住 这 个 连接 对 象 名 ， 下 面 还 要 使 用 它 ! 


4. 打开 数据 库 连接 
打开 数据 库 连接 使 用 的 是 数据 库 连接 对 象 的 Open 方法 完成 的 ， 具体 的 语法 形式 如 下 : 
连接 对 象 名 .open (); 
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这 里 ， 连 接 对 象 名 就 是 在 第 3 个 步骤 中 创建 的 数据 库 连 接 对 象 名 ， 例 如 : 之 前 创建 的 
conn。 那 么 ， 打 开 数 据 库 连接 conn 的 语句 就 如 下 : 


conn.Open (); 


5. 关闭 数据 库 连接 

关闭 数据 库 连接 要 在 对 数据 库 操作 后 使 用 ， 但 是 ， 一 定 要 注意 的 是 ， 只 要 在 程序 中 有 
数据 库 连 接 处 于 打开 状态 ， 就 要 在 对 其 操作 完成 后 关闭 。 如 果 不 注意 数据 库 连 接 的 关闭 ， 
数据 库 就 无 法 释放 数据 库 连接 ， 而 占用 数据 库 连 接 的 数量 。 关 闭 数 据 库 连接 使 用 Close 方 
法 来 完成 ， 具 体 的 语法 形式 如 下 : 

连接 对 象 名 .Close (); 

至 此 ， 就 完成 了 数据 库 连接 对 象 基 本 使 用 方法 的 学 习 。 在 后 面 的 数据 库 操作 中 ， 都 离 
不 开 该 对 象 的 使 用 。 
17.1.3 ”使 用 Command 操作 SQL Server 数据 库 

Command 类 是 用 来 操作 数据 库 中 的 数据 的 ， 但 是 在 使 用 Command 对 象 之 前 ， 首 要 的 
步骤 就 是 要 打开 数据 库 的 连接 。 与 Connection 类 一 样 ， 引 用 System.Data.SqlClient 命名 空 
间 ，Command 类 就 应 该 写成 SqlCommand。 使 用 Command 操作 SQL Server 数据 库 ， 分 为 
如 下 4 个 步骤 。 

1. 创建 数据 库 连 接 对 象 并 打开 数据 库 连 接 

使 用 上 一 小 节 的 方法 ， 创 建 并 打开 数据 库 连 接 ， 上 具体 的 语句 如 下 : 


SqlConnection conn =new SqlConnection (Server=.\MSSQLSERVER2008; 
database=chapter17; Integrated Security=True); // 创 建 连接 对 象 
conn.Open () ; // 打 开 数 据 库 连 接 


2. 创建 Command 对 象 

在 创建 Command 对 象 时 ， 通 常 需要 使 用 两 个 参数 ， 一 个 是 连接 对 象 名 ， 另 一 个 是 要 
执行 对 数据 表 操作 的 SQL 语句 。 创 建 Command 对 和 象 的 语法 形式 如 下 : 

SqlCommand command name = new SqlCommand (SQL, conn name); 
其 中 : 

口 command name: 命令 对 象 名 称 。 

口 SQL: 要 执行 的 SQL 语句 。 

口 conn_name: 连接 对 象 名 称 。 

使 用 上 面 的 语法 ， 创 建 Command 对 象 的 语句 如 下 : 

SqlCommand cmd = new SqlCommand (SQL, conn) 

这 里 ，cmd 是 命令 对 象 名 ，conn 是 连接 对 象 名 ，SQL 就 代表 了 要 执行 的 数据 库 操作 语 
句 。 在 实际 应 用 中 ， 要 在 该 语句 之 前 定义 具体 的 SQL 语句 。 例 如 : 查询 表 中 的 数据 、 向 表 
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中 添加 数据 等 。 
3. 执行 Command 对 象 中 的 SQL 语句 


在 执行 SQL 语句 时 ,通常 把 要 执行 的 SQL 语句 分 成 两 类 ， 一 类 是 执行 查询 的 SQL 语 
句 ， 一 类 是 执行 非 查询 的 SQL 语句 。 下 面 就 分 别 讲解 如 何 执行 这 两 类 SQL 语句 。 

(1) 执行 查询 的 SQL 语句 

所 谓 查询 的 SQL 语句 ， 就 是 以 SELECT 关键 字 来 编写 的 SQL 语句。 执行 查询 的 SQL 
语句 ， 使 用 Command 对 象 中 的 ExecuteReader 方法 ， 返 回 SqlDataReader 类 型 的 数据 。 具 
体 的 语法 形式 如 下 : 


SqlDataReader datareader name=command name .ExecuteReader (); 


比 中 ， 

口 datareader name: 数据 库 读 取 对 象 名 。 

口 command_name: 命令 对 象 名 。 

应 用 在 上 一 小 节 创建 的 命令 对 象 ， 执 行 查询 的 SQL 语句 ， 具 体 语 句 如 下 : 


SqlDataReader dr=cmd.ExecuteReader (); 


那么 ， 将 查询 出 的 数据 存储 到 了 数据 库 读 取 对 象 ， 如 果 查 看 到 该 对 象 中 的 值 呢 ?” 上 有 具体 
的 语句 如 下 : 

If (dr.Read ()) // 判 断 dr 中 是 否 存 在 查询 数据 

string str = dr[0] .toString() 7 // 取 查询 结果 中 第 1 行 第 1 列 的 数据 

其 中 ，dr[0] 中 的 0 还 可 以 换 成 是 表 中 的 具体 列 名 ， 第 1 列 的 编号 是 0。 另 外 ， 如 果 要 
查询 的 数据 不 仅 是 1 行 时 ， 还 可 以 将 证 语句 换 成 是 while， 这 样 就 可 以 将 数据 表 中 的 全 部 
数据 显示 出 来 了 。C# 语 言 中 的 让 和 while 语句 的 用 法 , 基本 与 SQL 语言 中 的 结构 控制 语句 
类 似 ， 这 里 就 不 再 多 说 了 。 

(2) 执行 非 查 询 SQL 语句 

所 谓 非 查询 的 SQL 语句 ,通常 就 是 对 数据 表 中 数据 的 添加 、 删 除 以 及 修改 的 操作 。 使 
用 Command 对 象 执行 非 查询 SQL 语句 用 的 是 ExecuteNonQuery 方法 。 该 方法 返回 的 是 一 
个 整数 类 型 的 数据 ， 当 返回 的 值 是 -1 时 ， 代 表 对 数据 表 操 作 失 败 ; 当 返 回 值 是 0 时， 代表 
没有 对 数据 表 中 的 数据 有 任何 影响 ， 当 返回 值 是 一 个 具体 的 整数 时 ， 代 表 对 数据 表 中 更 新 
的 数据 行 数 。 有 具体 的 语法 形式 如 下 : 


int returnvalue=command name. ExecuteNonQuery (); 


其 中 
口 returnvalue: 变量 名 。 用 于 接收 非 查询 方法 执行 后 返回 的 结果 。 
口 command_name: 命令 对 象 名 。 此 时 的 命令 对 象 必 须 是 执行 非 查询 语句 的 命令 对 象 。 


4. 关闭 数据 库 连 接 
当 完 成 了 对 数据 库 的 所 有 操作 后 ， 最 后 一 步 就 关闭 数据 库 连接 了 。 使 用 下 面 的 语句 即 
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可 完成 操作 。 
conn.Close (); 


至 此 ， 就 完成 了 使 用 Command 对 象 操作 数据 库 的 讲解 。 
17.1.4 使 用 DataSet 和 DataAdapter 操作 SQL Server 数据 库 


在 操作 SQL Server 数据 库 时 ， 使 用 DataSet 和 DataAdapter 基本 都 是 在 查询 数据 表 时 
使 用 的 。 下 面 就 来 见证 它们 的 使 用 方法 吧 ， 分 为 如 下 5 个 步骤 。 


1. 创建 数据 库 连 接 对 象 并 打开 数据 库 连 接 
与 使 用 Command 对 象 一 样 ， 第 1 步 都 是 要 创建 并 打开 数据 库 连 接 。 具 体 的 语句 如 下 : 


SqlConnection conn =new SqlConnection (Server=.\MSSQLSERVER2008; 
database=chapter17; Integrated Security=True); // 创 建 连接 对 象 
conn.Open(); // 打 开 数 据 库 连 接 


2. 创建 DataAdapter 对 象 


DataAdapter 类 在 引用 了 System.Data.SqlClient 后 ， 使 用 SqlDataAdapter 类 即 可 。 在 创 
建 DataAdapter 对 象 时 与 Command 对 象 类 似 ， 都 是 需要 两 个 参数 ， 一 个 是 要 执行 的 SQL 
语句 ， 一 个 是 连接 对 象 名 。 但 是 ， 这 里 需要 注意 的 是 SQL 语句 是 执行 查询 的 语句 。 具体 的 
语法 形式 如 下 : 


SqlDataAdapter DataAdapter name= new SqlDataAdapter (SELECT-SQL， 
conn_name); 


其 中 : 

口 DataAdapter name: 数据 适配器 对 象 名 称 。 

口 SELECT-SQL: 执行 查询 的 SQL 语句 。 

口 conn_name: 连接 对 象 名 。 

应 用 上 面 的 语句 ， 创 建 一 个 DataAdapter 对 象 ， 具 体 的 语句 如 下 : 


SqlDataAdapter ada= new SqlDataAdapter (SQL, conn); 
这 里 ，ada 是 DataAdapter 对 象 名 称 ，SQL 是 查询 语句 ，conn 是 连接 对 象 名 。 
3. 创建 DataSet 对 象 


DataAdapter 对 象 创 建 后 ， 还 要 创建 DataSet 对 象 用 以 存放 查询 结果 。 创 建 DataSet 对 
象 很 简单 ， 语 法 如 下 : 

DataSet dataset name= new DataSet (); 

这 里 ，dataset_name 是 数据 集 DataSet 对 象 的 名 称 。 

使 用 上 面 的 语句 ， 创 建 一 个 DataSet 对 象 ， 语 句 如 下 : 


DataSet ds= new DataSet (); 
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4. 将 数据 填充 到 DataSet 对 象 中 

将 DataAdapter 中 的 数据 填充 到 DataSet 对 象 中 使 用 的 是 DataAdapter 的 Fill 方法 完成 
的 。 具 体 的 语法 形式 如 下 : 

DataAdapter name.Fill (DataSet name); 

其 中 : 

口 DataAdapter name: 数据 适配器 名 称 。 

口 DataSet_name: 数据 集 名称 。 

应 用 上 面 的 语法 填充 之 前 创建 的 数据 集 对 象 ， 语 句 如 下 : 


ada.Fill (ds); 


5. 关闭 数据 库 连接 

当 完 成 了 对 数据 集 的 操作 后 ， 就 可 以 关闭 数据 库 连接 了 。 具 体 的 语句 如 下 : 
conn.Close (); 

至 此 ， 使 用 DataAdapter 和 DataSet 查询 数据 的 操作 就 完成 了 。 


全 说 明 : DataSet 的 使 用 是 非常 灵活 的 ， 可 以 得 到 全 部 的 数据 ， 也 可 以 通过 全 部 数据 得 到 
其 中 的 某 些 数据 。 如 果 想 进一步 学 习 DataSet 的 使 用 ， 可 以 参考 相关 的 C# 书 籍 。 


17.2 使 用 Windows 窗 体 程序 完成 文章 管理 系统 


通过 前 面 学 习 过 的 SQL 语句 以 及 本 章 中 简单 讲解 的 ADO.NET 中 五 大 类 的 使 用 方法 ， 
就 可 以 使 用 C# 语 言 编 写 一 些小 功能 了 。 在 本 节 中 就 将 带领 读者 使 用 C# 语 言 完成 一 个 简单 
的 Windows 窗 体 程 序 ， 用 以 完成 文章 信息 管理 的 功能 。 


17.2.1 Windows 窗 体 程序 的 开发 环境 介绍 


所 谓 Windows 窗 体 程序 就 是 类 似 于 我 们 正在 使 用 的 Windows 操作 系统 一 样 的 程序 ， 
都 是 以 窗 体 的 形式 来 显示 数据 的 。 在 本 章 中 开发 Windows 窗 体 程序 使 用 的 是 Visual Studio 
C# 2010 学 习 版 。 该 版 本 可 以 在 微软 网 站 上 直接 下 载 并 免费 试用 。 该 版 本 程序 打开 后 的 界 
面 如 图 17.1 所 示 。 

当然 ， 还 有 其 他 的 版 本 可 以 下 载 ， 有 兴趣 的 读者 可 以 试用 一 下 。 

既然 要 使 用 该 软件 开发 Windows 窗 体 程序 ， 那 么 ， 就 要 知道 Windows 窗 体 究竟 是 什 
么 样 的 。 下 面 就 来 一 同 创建 一 个 Windows 窗 体 程 序 并 认识 其 开发 界面 。 


1. 创建 Windows 窗 体 程序 


在 图 17.1 所 示 界 面 中 ， 单 击 菜单 “文件 ”|“ 新 建 项 目 ” 选 项 ， 弹 出 图 17.2 所 示 的 
界面 。 
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图 17.2 新建 项 目 界面 


在 图 17.2 所 示 界 面 中 ， 选 择 “Windows 窗 体 应 用 程序 ”并 给 其 起 个 名 字 即 可 。 单 击 
“确定 ”按钮 ， 就 完成 了 一 个 Windows 窗 体 程序 的 创建 。 效 果 如 图 17.3 所 示 。 


2. 认识 Windows 窗 体 程序 界面 


在 图 17.3 所 示 界 面 中 ， 读 者 已 经 看 到 了 Windows 窗 体 程序 的 设计 页 面 究竟 是 什么 样 
了 。 那 么 ， 下 面 就 来 具体 说 说 该 界面 中 各 窗口 的 作用 。 
口 工具 箱 : 用 来 存放 放置 到 窗 体 上 的 控件 。 单 击 该 工具 箱 中 任意 控件 直接 拖 搜 到 窗 
体 上 即 可 。 
口 窗 体 界面 : 工具 箱 右边 的 部 分 。 它 就 是 用 户 要 设计 和 使 用 的 窗 体 了 。 
口 解决 方案 资源 管理 器 : 用 来 显示 该 项 目 中 的 文件 构成 。 
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图 17.3 第 一 个 Windows 窗 体 程序 
口 属性 窗口 : 用 来 设置 拖 搜 到 窗 体 上 的 控件 属性 ， 包 括 : 控件 中 显示 的 文本 、 颜 色 
和 大 小 等 信息 。 同 时 ， 还 可 以 在 该 窗口 中 设计 该 页 面 中 控件 执行 的 事件 ， 比 如 : 
单 击 事件 、 双 击 事件 等 。 
17.2.2 ”数据 表 的 设计 
在 本 章 中 着 重 讲解 的 文章 管理 系统 中 ， 主 要 完成 的 功能 是 对 文章 信息 的 添加 、 修 改 、 
删除 以 及 查询 。 文 章 信息 主要 包括 文章 编号 、 文 章 题 目 、 摘 要 和 作者 等 信息 。 有 具体 的 表 结 
构 如 表 17-1 所 示 。 


表 17-1 文章 信息 表 (paperinfo) 


序号 列 名 数据 类 型 说 明 
1 id int 编号 
2 title varchar(50) 题目 
3 author decimal(10) 作者 
4 company varchar(50) 单位 
5 abstract varchar(500) 摘要 
6 keyword varchar(50) 关键 词 
7 pagecount int 页 数 


读者 可 以 根据 上 面 的 表 结 构 在 SQL Server 中 创建 数据 表 paperinfo, 相信 读者 对 建 表 操 
作 已 经 不 再 陌生 了 。 另 外 ， 在 创建 数据 表 之 前 先 创建 一 个 本 章 使 用 的 数据 库 chapter17。 企 
业 管 理 器 中 文章 信息 表 设 计 效 果 如 图 17.4 所 示 。 

除了 SQL Server 工具 可 以 创建 数据 表 外 ， 还 可 以 直接 使 用 Visual C# 工 具 来 连接 数据 
库 ， 因 为 它们 出 自 于 同一 个 公司 嘛 。 下 面 就 来 介绍 如 何 使 用 Visual C# 中 的 数据 库 资源 管 
理 器 。 
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1. 打开 数据 库 资 源 管 理 器 


在 图 17.3 所 示 的 菜单 栏 中 依次 选择 “视图 ”|“ 其 他 窗口 ”| “数据 库 资源 管理 器 ” 选 
项 ， 出 现 如 图 17.5 所 示 的 界面 。 


、 Microsoft SQL Server Management SG =|Djxl 
文件 (编辑 (E) 视图 W。 项 目 P) 调试 D) 表 设 计 器 (U) 
工具 (D 窗口 Ww) 社区 人 四。 帮助 册 

:也 新 建 查询 六 | 古人 六 钙 | 刘 区 昌 总 5 

和 || 污 间 岂 同 加 六 名 


WIDW-9527\MS.-dbo.paperinfo| 
列 名 


| 
oe 
| 


图 17.4 文章 信息 表 在 企业 管理 器 中 的 创建 效果 图 17.5 数据 库 资源 管理 器 界面 


2. 创建 数据 库 连接 

在 图 17.5 所 示 界 面 中 , 右 击 “数据 连接 ”选项 , 在 弹出 的 右键 菜单 中 选择 “添加 连接 ” 
选项 ， 出 现 如 图 17.6 所 示 的 界面 。 

在 图 17.6 所 示 界 面 中 ， 选 择 “Microsoft SQL Server 数据 库 文件 ”选项 ， 单 击 “ 继 续 ” 
按钮 ， 出 现 添加 连接 界面 ， 如 图 17.7 所 示 。 


可 对 
第 入 信息 以 这 接 到 定 的 数据 源 ， 或 单 击 "更 改选 择 另 一 
个 数 指 源 和 琴 提 供 程序。 


数据 源 (5): 
JMierosoft 5QL Server 妆 据 库 文件 (5qcien 。 更 By(C)， 


数据 库 文件 名 (新 建 或 现 有 名 称 XD): 


LServer Compact 3.5 
Microsoft 5QL 5erver 数据 库 文件 个 合用 5QL Server 身份 验证 (Q) 

朋 Ps( 

SP) 


全 1 


高 级). 
测 动 和 接 人 D 取消 


图 17.6 ”选择 数据 源 界面 图 17.7 添加 连接 界面 


这 里 ， 选 择 之 前 创建 的 chapter17 数据 库 ， 并 单 击 “ 测 试 连接 ”按钮 ， 如 果 测 试 成 功 ， 
则 单 击 “确定 ”按钮 ， 即 可 创建 好 与 数据 库 chapter17 中 的 连接 ， 如 图 17.8 所 示 。 

至 此 ， 就 将 Visual C# 与 数据 库 chapter17 连接 起 来 了 。 以 后 读者 就 可 以 在 这 个 工具 下 
操作 数据 表 。 操 作 方法 与 在 企业 管理 器 中 操作 的 方法 类 似 ， 这 里 就 不 再 袭 述 了 。 
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17.2.3 添加 文章 功能 的 实现 
在 开始 开发 文章 管理 功能 之 前 ， 先 将 解决 方案 中 的 文件 添加 好 。 在 本 系统 中 ， 解 决 方 
案 中 的 文件 列表 如 图 17.9 所 示 。 


各 
ed 
EE 
|| 
岗 


13 回 | 日 国 
司 解 岂 方 案 *chapter17 人 1 个 项 目 ) 


图 17.8 ”连接 chapter17 的 效果 图 17.9 解决 方案 中 的 文件 列表 


从 图 17.9 中 ， 可 以 看 到 在 解决 方案 chapter17 下 共有 3 个 窗 体 文件 ， 作 用 如 下 。 

口 AddPaper.cs: 用 于 添加 文章 信息 。 

口 Modify.cs: 用 于 修改 文章 信息 。 

口 QueryAndDel: 用 于 查询 和 删除 文章 信息 。 

在 本 小 节 中 ， 将 带领 读者 完成 文章 管理 功能 中 的 第 一 个 功能 ， 即 添加 文章 。 完 成 该 功 
能 需要 经 过 页 面 设计 和 代码 编写 两 个 步骤 。 


1. 添加 文章 界面 设计 


添加 文章 的 界面 设计 主要 考虑 的 是 对 表 17-1 中 的 内 容 添 加 。 在 表 17-1 中 ， 只 有 文章 
编号 不 用 添加 ， 使 用 SQL Server 的 自 增长 序列 即 可 完成 ， 其 他 的 字段 都 需要 添加 。 在 添加 
信息 时 使 用 工具 箱 里 面 的 文本 框 、 标 签 以 及 多 行文 本 框 等 控件 完成 。 界 面 如 图 17.10 所 示 。 

了 5 可 

三 文章 信息 
是 | 
二 厂 
mn 


| 


we | _ | 
图 17.10 ”添加 文章 界面 


-ns 


2 
在 


该 功能 


09 
10 
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添加 文章 功能 的 代码 


图 17.10 所 示 界 面 中 ， 填 写 完 信息 后 ， 单 击 “ 确 定 ”按钮 ， 即 完成 数据 添加 的 功能 。 
使 用 的 就 是 按钮 的 单 击 事件 ， 并 在 该 单 击 事件 中 加 入 如 下 代码 。 

/// <summary> 

/// 添加 文章 信息 


/// </summary> 
Private void buttonl Click(object sender, EventArgs e) 


{ 
/ /创建 数据 库 连 接 
SqlConnection conn= new 
SqlConnection (@"Server=.\MSSQLSERVER2008; database=chapter17; 
Integrated 


Security=True"); 
try 
{ 
conn.Open () ; // 打 开 数据 库 连接 
string sql = "INSERT INTO paperinfo VALUES('{0}', {1}','{2}','{3}', 
和 人 // 编 写 SQL 语句 
sql = string.Format (Sq1，textBox1l .Text，textBox2 .Text，textBox3.Text， 
richTextBoxl.Text, textBox5.Text, int.Parse (textBox4.Text) ) 
// 格 式 化 SQL 语句 
SqlCommand cmd = new SqlCommand (sql，conn); // 创 建 命令 对 象 
int returnvalue = cmd.ExecuteNonQuery (); // 执 行 SQL 语句 
if (returnvalue!=-1) // 判 断 是 否 添加 成 功 
{ 
MessageBox.Show ("添加 成 功 !"); 
. 
| 
catch 
R 
MessageBox .Show ("操作 错误 !") ; 
' 
finally 
conn.Close (); // 关 闭 数据 库 连 接 
} 
} 
中 : 
第 07 一 09 行 : 创建 数据 库 连接 。 


第 12 行 : 打开 数据 库 连 接 。 
第 13 一 14 行 : 编写 SQL 语句 并 填充 占 位 符 。 
第 15 行 : 创建 命令 对 象 。 
第 16 行 : 执行 添加 的 SQL 语句 。 
第 17 一 20 行 : 判断 是 否 添加 成 功 。 
第 24 行 : 当前 面 的 添加 语句 出 现 异 常 时 ， 弹 出 添加 错误 的 提示 。 在 C# 语 言 中 捕 
获 异常 的 语句 是 try...catch...finally。 
第 28 行 : 关闭 数据 库 连接 。 
行 窗 体 ， 添 加 文章 的 效果 如 图 17.11 所 示 。 
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Be | 
图 17.11 添加 文章 功能 
至 此 ， 添 加 文章 的 功能 就 完成 了 ， 读 者 可 以 在 数据 库 中 看 看 数据 是 否 真 的 加 入 到 文章 
信息 表 中 了 。 
17.2.4 查询 文章 功能 的 实现 


有 了 添加 文章 功能 实现 的 基础 ， 读 者 应 该 对 窗 体 的 操作 很 熟悉 了 ， 下 面 就 来 完成 查询 
文章 的 功能 。 查 询 文章 功能 仍然 分 为 两 个 部 分 ， 一 个 是 界面 设计 ， 另 一 个 是 代码 编写 。 
1. 查询 文章 功能 的 界面 


在 本 系统 中 查询 文章 功能 只 提供 对 文章 名 称 一 个 条 件 的 查询 。 该 界面 可 以 通过 工具 箱 
中 的 标签 、 文 本 框 、 按 钮 以 及 数据 控件 DataGridView 来 完成 设计 。 界 面 如 图 17.12 所 示 。 


| 
文章 名 称 _ | WM | 


图 17.12 查询 文章 界面 设计 
这 里 将 删除 功能 也 放 到 了 该 界面 中 实现 ， 因 此 ， 在 界面 中 多 了 一 个 “删除 ”按钮 。 
2. 查询 文章 功能 的 代码 编写 
在 图 17.12 所 示 的 界面 中 ， 要 添加 两 部 分 的 代码 ， 一 部 分 是 在 窗 体 加 载 时 显示 所 有 的 
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文章 信息 ; 另 一 部 分 是 在 单 击 “ 查 询 ” 按 钮 和 时， 显示 查询 结果 。 
(1) 窗 体 加 载 事件 的 代码 
在 图 17.12 所 示 界 面 中 的 窗 体 加 载 事 件 (Load) 中 ， 加 入 如 下 代码 。 


01 private void Form2 Load (object sender, EventArgs e) 


02 
03 
04 
05 


1 
// 创 建 数据 库 连接 
SqlConnection conn= new 


SqlConnection (@"Server=.\MSSQLSERVER2008; database=chapter17; 
Integrated 


06 Security=True"); 
07 try 
08 { 
09 conn.Open () ; // 打 开 数 据 库 连 接 
10 string sql = "select id as ' 编 号 ',title as ' 题 目 ',author as ' 作 者 '， 
keyword as ' 关 键 词 ' from 1lpaperinfo";  // 编 写 语句 
12 SqlDataAdapter ada = new SqlDataAdapter(sql, conn) 
// 创 建 数据 适配器 对 象 
13 DataSet ds = new DataSet(); // 创 建 数据 集 对 象 
14 ada.Fill(ds); // 填 充 数 据 集 
15 dataGridViewl.DataSource = ds.Tables[0]; 
// 将 数据 集中 的 内 容 与 datagridview 绑 定 
16 | 
hy catch 
18 { 
19 MessageBox .Show ("操作 错误 !"); 
20 } 
总和 finally 
{ 
23 conn.Close (); // 关 闭 数据 库 连 接 
24 } 
25 l 
其 中 : 
口 第 03 一 06 行 : 连接 数据 库 。 
口 第 09 行 : 打开 数据 库 连 接 。 
口 第 10~11 行 : 编写 SQL 语句 ,查询 出 全 部 数据 。 在 查询 语句 中 使 用 了 别名 使 查询 
结果 中 显示 中 文 列 名 。 
口 第 12 行 : 创建 数据 适配器 对 象 。 
口 第 13 行 : 创建 数据 集 对 象 。 
口 第 14 行 : 填充 数据 集 。 
口 第 15 行 : 将 数据 集中 的 结果 绑 定 到 DataGridView 控件 中 。DataGridView 控件 是 
在 查询 功能 中 经 常 使 用 的 一 个 数据 控件 。 
口 第 19 行 : 当前 面 的 内 容 出 现 异常 时 ， 弹 出 操作 错误 的 提示 。 
口 第 23 行 : 关闭 数据 库 连 接 。 


(2)“ 查 询 ” 按 钮 的 单 击 事件 的 代码 
在 图 17.12 所 示 界 面 中 ， 输 入 文章 名 称 ， 单 击 “ 查 询 ” 按 钮 就 可 以 在 DataGridView 中 
查询 到 结果 。 在 “查询 ”按钮 的 单 击 事件 中 加 入 的 代码 如 下 : 


01 private void buttonl Click(object sender, EventArgs e) 


02 


le 
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03 // 创 建 数据 库 连 接 


04 SqlConnection conn= new 
05 SqlConnection (@"Server=. \MSSQLSERVER2008; database=chapter17; 


Integrated 

06 Security=True"); 

07 try 

08 

09 conn.Open(); // 打 开 数 据 库 连接 

10 string sql = "select id as ' 编 号 ', title as 题目" ,author as ' 作 者 '， 

keyword as' 关 键 词 ' from 11paperinfo where title='"+textBoxl.Text+"'"; 

// 编 写 语句 

12 SqlDataAdapter ada = new SqlDataAdapter(sql, conn); 
// 创 建 数据 适配器 对 象 

13 DataSet ds = new DataSet() // 创 建 数据 集 对 象 

14 ada.Fill(ds); // 填 充 数 据 集 


15 dataGridViewl.DataSource = ds.Tables[0]; 
// 将 数据 集中 的 内 容 与 datagriqdview 绑 定 


16 1 

下 catch 

18 { 

19 MessageBox .Show(" 操 作 错误 !") ; 

20 1 

21 finally 

2 

23 conn.Close () // 关 闭 数据 库 连接 
24 1 

25 } 


读者 观察 上 面 的 代码 会 发 现 与 在 窗 体 加 载 事件 中 编写 的 代码 几乎 是 一 样 的 ， 只 是 查询 
时 使 用 的 SQL 语句 不 同 而 已 。 

至 此 ， 查 询 文章 功能 就 基本 完成 了 ， 下 面 就 是 验证 劳动 成 果 的 时 候 了 。 查 询 界面 如 
图 17.13 所 示 。 


图 17.13 ”查询 文章 功能 的 效果 
外 说 明 : 在 本 查询 中 , 使 用 的 是 等 值 查询 , 也 可 以 使 用 like 语句 对 文章 信息 进行 模糊 查询 。 


17.2.5 删除 文章 功能 的 实现 
删除 文章 的 功能 是 与 查询 文章 的 功能 在 同一 个 文件 中 完成 的 ， 因 此 ， 就 不 再 讲解 其 界 
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面 的 设计 了 。 删 除 文章 功能 的 操作 是 通过 单 击 选择 一 条 文章 信息 ， 然 后 单 击 “ 删 除 ” 按 钮 


完成 删除 操作 的 。 因 此 ， 只 需要 在 “删除 ”按钮 的 单 击 事件 中 加 入 代码 即 可 。 代 码 如 下 : 

01 /// <summary> 

02 /// 删除 选中 行 的 文章 信息 

03 /// </summary> 

04 private void button2 Click(object sender, EventArgs e) 

05 

06 // 创 建 数据 库 连 接 

07 SqlConnection conn= new 

08 SqlConnection (@"Server=.\MSSQLSERVER2008; database=chapter17; 
Integrated 

09 Security=True") 

10 try 

Dl { ”// 获 取 选 中 行 id 列 中 的 值 

2 int id = int.Parse(dataGridView1l.SelectedRows [0] . 

Cells[0] .Value.ToString ()); 
13 conn.Open (); // 打 开 数 据 库 连接 
14 string sql = "delete from paperinfo where id="+id; 
// 编 写 SQL 语句 

15 SqlCommand cmd = new SqlCommand (sql，conn); // 创 建 命令 对 象 

16 int returnvalue = cmd.ExecuteNonQuery (); // 执 行 SQL 语句 

本! if (returnvalue! = -1) // 判 断 是 否 删 除 成 功 

18 { 

19 MessageBox.Show ("删除 成 功 !")， 

20 ， 

2 } 

4 catch 

23 { 

24 MessageBox .Show ("操作 错误 !") ; 

25 } 

26 finally 

Eh { 

28 conn.Close () ; // 关 闭 数据 库 连接 

29 

30 } 

其 中 ， 

口 第 06 一 09 行 : 创建 数据 库 连 接 。 

口 第 12 行 : 获取 选中 行 的 id 列 的 值 。 

口 第 13 行 : 打开 数据 库 连 接 。 

口 第 14 行 : 编写 SQL 语句 。 

口 第 15 行 : 创建 命令 对 象 。 

口 第 16 行 : 执行 删除 的 SQL 语句 。 

口 第 17 一 20 行 : 判断 是 否 删除 成 功 。 

口 第 24 行 :当前 面 的 添加 语句 出 现 异常 时 ， 弹 出 操作 错误 的 提示 。 

口 第 28 行 : 关闭 数据 库 连 接 。 


至 此 ， 删 除 文章 的 功能 就 完成 了 。 
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修改 文章 功能 


掌握 了 前 面 对 文 章 的 添加 和 查询 功能 ， 完 成 修改 文章 的 功能 就 是 小 菜 一 碟 了 。 本 系统 
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中 修改 文章 首先 将 要 修改 的 文章 查询 出 来 ， 然 后 再 对 其 文章 内 容 进 行 修改 。 因 此 ， 本 功能 
分 如 下 3 个 步骤 完成 。 


1. 修改 功能 界面 设计 


修改 功能 的 界面 与 增加 文章 功能 的 界面 类 似 ， 只 不 过 增加 了 查询 的 文本 框 和 按钮 。 界 
面 如 图 17.14 所 示 。 


I 
人 | 


图 17.14 ”修改 文章 功能 界面 


2. 根据 编号 查询 功能 代码 编写 


在 图 17.14 所 示 界 面 中 ， 通 过 单 击 “ 查 询 ” 按 钮 ， 查 询 出 该 编号 所 对 应 的 文章 信息 。 
代码 如 下 : 


01 private void buttonl Click(object sender, EventArgs e) 


02 
03 // 创 建 数据 库 连 接 
04 SqlConnection conn = new SqlConnection(@"Server=.\MSSQLS 
ERVER2008;database=chapterl7;Integrated Security=True") 
05 try 
06 { 
07 conn.Open(); // 打 开 数 据 库 连 接 
08 string sql = "select title,author, keyword,abstract, 
Ppagecount, company from paperinfo where id="+int.Parse 
(textBox6.Text); // 编 写 SQL 语句 
09 sql = string.Format (sql，textBoxl.Text); ”// 填 充 SoL 语句 
10 SqlDataAdapter ada = new SqlDataAdapter (sql, conn); 
// 创 建 数据 适配器 对 象 
5 DataSet ds = new DataSet (); // 创 建 数据 集 对 象 
12 ada.Fill (ds); // 填 充 数 据 集 
13 textBoxl.Text = ds.Tables[0] .Rows[0] ["title"] .ToString(); 
// 向 文本 框 赋值 
14 textBox2 .Text = ds.Tables[0] .Rows[0] ["author"] . 


ToString(); 
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下 5 textBox3 .Text = ds.Tables[0] .Rows[0] ["company"] . 
ToString(); 

16 textBox4.Text = ds.Tables[0] .Rows[0] ["pagecount"]. 
ToString()» 

7 richTextBoxl.Text = ds.Tables[0] .Rows[0] ["abstract"]. 
ToString()? 

18 textBox5 .Text = ds.Tables[0] .Rows [0] ["keyword"]. 
ToString() 

19 } 

20 catch 

21 { 

本 MessageBox .Show (" 操 作 错 误 ") ; 

23 } 

24 finally 

2 { 

26 conn.Close (); // 关 闭 数据 库 连 接 

27 } 

28 } 

其 中 : 


口 第 04 行 : 创建 数据 库 连接 。 

口 第 07 行 : 打开 数据 库 连 接 。 

口 第 08 一 09 行 : 编写 SQL 语句 并 填充 占 位 符 的 值 。 

口 第 10 行 : 创建 数据 适配器 对 象 。 

口 第 11 一 12 行 : 创建 并 填充 数据 集 对 象 。 

口 第 13 一 18 行 : 将 数据 集中 的 值 显示 在 对 应 的 文本 框 中 。 

口 第 22 行 ， 当 前 面 的 操作 出 现 异 常 时 ， 弹 出 “操作 错误 ”的 提示 框 。 
口 第 26 行 : 关闭 数据 库 连 接 。 


3. 修改 文章 信息 的 代码 编写 


在 图 17.14 所 示 界 面 中 ， 根 据 文章 编号 将 文章 信息 查询 出 来 ， 修 改 信息 后 单 击 “ 确 定 ” 
按钮 即 可 更 新 文章 信息 。 在 “确定 ”按钮 的 单 击 事件 中 加 入 如 下 代码 。 


01 /// <summary> 

02 /// 添加 文章 信息 

03 /// </summary> 

04 private void buttonl Click(object sender, EventArgs e) 
05 { 

06 // 创 建 数据 库 连 接 


07 SqlConnection conn= new 

08 SqlConnection (@"Server=.\MSSQLSERVER2008; database=chapter17; 
Integrated 

09 Security=True"); 

10 try 

于 人 { 

了 2 conn.Open () ; // 打 开 数据 库 连接 

13 string sql = "UPDATE paperinfo SET 

14title=' {0}',author='{1}',company="'{2}',pagecount={3},abstract=' {4}', 
keyword='{5}'where id={6}"; 

15 sql= string.Format (sql, textBoxl.Text, textBox2.Text, textBox3.Text, 

16int.Parse (textBox4.Text) ,richTextBox1l.Text，textBox5 .Texty 
int.Parse (textBox6.Text) ) 

ry SqlCommand cmd = new SqlCommand (sql，conn); // 创 建 命令 对 象 

18 int returnvalue = cmd.ExecuteNonQuery (); // 执 行 SQL 语句 


"Se 


第 5 篇， 数据 库 的 应 用 


19 if (returnvalue!=-1) // 判 断 是 否 添加 成 功 
20 

2 MessageBox .Show ("添加 成 功 ! ") ; 
2 } 

23 四 

24 catch 

25 { 

26 MessageBox .Show (" 操 作 错 误 ! ") ; 

人 } 

28 finally 

29 

30 conn.Close (); // 关 闭 数据 库 连 接 
31 直 

32 } 


读者 阅读 了 上 面 这 么 多 代码 ， 一 定 会 有 一 些 似曾相识 的 感觉 。 没 错 ， 上 面 的 代码 与 前 
面 的 添加 文章 信息 的 代码 比较 ， 只 是 第 13 一 16 行 的 SQL 语句 有 变化 ， 其 他 的 都 一 样 。 
经 过 了 前 面 3 个 步骤 的 操作 ， 现 在 就 可 以 看 看 效果 了 ， 如 图 17.15 所 示 。 
SE 
FP | 
是 目 。 任 FTordTRZE 


作者 内 而 
单位 | 南京 大 学 页 数 hz 


甘 要 [村 广 讲 和解 了 加 向 使 用 Jurs 开 发 小 说 系统 
x 

修改 成 功 
确定 


.| 
图 17.15 ”修改 文章 功能 的 效果 


17.3 本 章 小 结 


在 本 章 中 主要 讲解 了 如 何 使 用 ADO.NET 连接 SQL Server 数据 库 以 及 应 用 ADO.NET 
开发 文章 管理 的 一 些 简 单 功能 。 其 中 ， 重 点 学 习 了 ADO.NET 中 的 五 大 类 即 Connection、 
Command、DataSet、DataAdapter 和 DataReader。 相 信 读 者 通过 本 章 的 学 习 ， 也 能 够 使 用 
ADO.NET 完成 一 些 简单 的 数据 操作 功能 。 
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JSP 使 得 Java 语言 在 Web 方向 的 应 用 更 加 便捷 和 广泛 , 利用 JSP 可 以 方便 地 把 从 数据 
库 中 获取 的 数据 在 页 面 进行 展示 , 同时 Java 语言 也 可 以 轻松 地 从 JSP 页 面 获取 数据 并 存储 
到 数据 库 中 。Java 要 想 高 性 能 地 操作 SQL Server 数据 库 ， 则 需要 使 用 SQL Server 官方 提供 
的 驱动 程序 。 本 章 将 利用 一 个 简单 的 在 线 订 购 系统 介绍 Java 如 何 操作 SQL Server 数据 库 。 

本 章 的 主要 知识 点 如 下 : 

口 了 解 B/S 结构 的 优势 

口 如 何 创建 数据 库 连 接 

口 如 何 配置 连接 池 

口 利用 JSP 和 Java 实现 订购 系统 


18.1 了 解 B/S 结构 的 优势 


B/S 结构 系统 方便 用 户 使 用 ， 服 务 端 需要 一 个 服务 器 而 客户 则 仅仅 需要 一 个 浏览 器 ， 
这 种 结构 相对 C/S 来 说 更 能 获得 用 户 的 青睐 。Java 语言 对 B/S 结构 的 开发 则 更 有 自己 的 优 
势 。 本 节 将 介绍 B/S 结构 的 特点 以 及 优势 。 


18.1.1 了 解 B/S 结构 的 优势 


B/S 结构 即 浏览 器 /服务 器 模式 , 它 是 Web 兴起 后 的 网 络 模式 ， 客 户 端 只 需要 一 个 浏览 
器 ， 绝 大 部 分 系统 功能 由 服务 器 完成 。 相 比较 C/S 模式 而 言 ， 由 于 网 页 浏览 器 的 普遍 使 用 ， 
使 得 这 种 B/S 模式 更 受 客户 欢迎 ， 因 为 客户 端 只 需要 浏览 器 就 可 以 完成 所 有 的 业务 操作 。 
下 面 简单 列举 了 浏览 器 /服务 器 模式 的 一 些 优点 。 


1. 客户 端 使 用 广泛 

B/S 模式 中 ， 浏 览 器 作为 客户 端 ， 当 今 网 络 发 展 迅 速 ， 各 种 植 入 网 页 浏览 器 的 设备 举 
不 胜 举 ， 因 此 以 浏览 器 作为 客户 端 是 最 明智 的 选择 。 

2. 有 效 减 少 客户 端 维护 工作 量 

C/S 模式 除了 服务 器 外 ， 客 户 端 也 需要 工作 人 员 安 装 软件 ， 因 此 客服 支持 人 员 不 得 不 
把 维护 客户 端 也 纳入 工作 一 部 分 。 而 B/S 模式 中 ， 只 要 网 页 浏览 器 不 出 问题 ， 那 么 就 不 会 
影响 客户 对 软件 的 操作 。 
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3. 系统 可 扩展 性 增强 ， 降 低 开发 商 成 本 


C/S 模式 中 ， 假 如 软件 升级 ， 通 常 都 是 服务 端 和 客户 端 同时 升级 。 而 使 用 B/S 模式 ， 
软件 升级 时 ， 客 户 端 基本 不 做 任何 修改 ， 修 改 的 只 是 服务 器 端 ， 这 样 当 软 件 功能 需求 发 
变化 时 , 开发 者 只 需要 在 服务 器 端 进行 修改 即 可 , 这 样 就 很 有 效 地 降低 了 开发 商 的 成 本 。 

4. B/S 结构 降低 客户 使 用 成 本 

客户 端 使 用 网 页 浏览 器 ， 通 常 不 需要 客户 对 其 进行 复杂 操作 就 能 使 用 软件 ， 还 有 就 是 
由 于 客户 端 只 需要 浏览 器 ， 那 么 这 就 降低 了 对 客户 端的 硬件 要 求 ， 同 时 也 降低 客户 使 用 
成 本 。 
全 注意 : B/S 模式 有 很 多 优势 ， 包 括 便捷 性 和 成 本 控制 方面 ， 但 这 并 不 意味 着 它 可 以 完全 
取代 C/S 模式 ， 每 个 模式 都 有 其 擅长 的 领域 ， 例 如 C/S 模式 就 更 擅长 游戏 和 杀毒 
软件 等 。 


卜 


| 让 


18.1.2 了 解 Java Web 服务 器 


Web 程序 在 运行 的 时 候 基 本 都 需要 一 个 容器 ， 编 译 后 的 程序 在 该 容器 中 运行 ， 它 为 程 
序 运行 提供 必要 的 环境 ， 这 个 容器 就 是 服务 器 。 例 如 ASP 的 服务 器 可 以 是 IS、PHP 服务 
器 可 以 是 Apache， 而 Java 也 有 它 自 己 特有 的 服务 器 。 

Java Web 程序 经 常 使 用 的 服务 器 有 Tomcat、Jboss、WebLogic 以 及 Websphere， 其 中 
WebLogic 以 及 Websphere 适合 大 型 商业 项 目 ， 而 平时 个 人 开发 使 用 较 多 的 是 Tomcat。 

Tomcat 是 免费 服务 器 ， 适 合 个 人 开发 以 及 调试 ， 它 是 Apache Jakarta 软件 组 织 的 一 个 
子 项 目 ， 是 在 SUN (已 被 Oracle 收购 ) 公司 的 JSWDK (Java Server Web Development Kit) 
基础 上 发 展 起 来 的 一 个 JSP 和 Servlet 规范 的 标准 实现 , 它 主要 作用 就 是 当 客 户 端 发 送 请 求 
过 来 时 ， 服 务 器 对 该 请 求 进行 处 理 ， 并 把 处 理 结果 返回 给 客户 端 。 

发 展 到 今天 ，Tomcat 除了 是 JSP 和 Servlet 规范 的 标准 实现 外 ， 也 被 用 于 了 商业 项 目 
中 ， 当 然 ， 它 更 适合 较 小 的 项 目 。 本 章 将 介绍 如 何在 该 服务 器 下 运行 一 个 简单 的 在 线 订 购 
系统 。 


18.2 在线 订购 系统 模块 设计 


Java 连接 SQL Server 数据 库 不 需要 过 多 的 操作 , 只 需要 不 多 的 代码 就 能 够 获得 数据 库 
连接 ， 但 为 了 保证 连接 的 高 性 能 ， 需 要 使 用 数据 库 开 发 商 提供 数据 库 驱动 包 。 在 线 订购 系 
统 详细 地 介绍 了 如 何 操作 数据 库 ， 本 节 将 介绍 在 线 订 购 系统 的 模块 划分 。 


18.2.1 订购 系统 的 流程 


在 线 订购 系统 相对 简单 ， 包 括 登 录 、 订 购 、 购 买 和 查看 数据 等 功能 ， 主 要 实现 在 线 订 
购 商 品 功能 , 使 用 者 可 以 方便 地 在 网 络 上 浏览 和 订购 物品 。 但 该 系统 不 是 一 个 完善 的 系统 ， 
只 是 方便 介绍 Java 如 何 操作 数据 库 数 据 。 有 关 在 线 订购 系统 的 流程 图 请 参考 图 18.1。 
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18.2.2 ”模块 介绍 Cm 7) 


一 个 软件 如 同一 家 企业 ， 每 家 企业 包含 多 个 部 门 ， 软 | 
件 也 一 样 ， 一 个 软件 道 常 包含 多 个 模块 ， 每 个 模块 实现 一 | 
定 的 业务 功能 ， 从 而 组 成 一 个 完整 的 系统 。 好 的 程序 模块 
之 间 的 关联 相对 较 少 ， 这 就 是 降低 了 模块 间 的 耦合 度 ， 低 -< 人 9 全 
耦合 对 程序 的 扩展 有 着 积极 的 作用 ; 同时 程序 模块 化 对 保 
证 程序 开发 进度 也 有 着 积极 作用 。 在 线 订购 系统 是 个 简单 [EL 
的 程序 ， 根 据 需求 可 以 分 成 以 下 4 个 模块 。 


和 让 归 吕 基 基 | | 一 ,型 险 订购 信 息 | 
该 模块 允许 已 经 存在 的 用 户 登 录 系 统 , 其 中 admin 是 。 [ws 

个 特殊 的 用 户 ， 该 用 户 可 以 查看 用 户 订单 。 其 他 还 有 test | 

用 户 ， 可 以 测试 功能 。 这 些 用 户 已 经 存 入 数据 库 当中 。 ES 
2， 商 品 浏览 模块 一 


该 模块 主要 对 系统 中 已 经 存在 的 商品 进行 浏览 ， 用 户 
订购 的 商品 需要 从 这 里 面 选取 ,有关 商 品 数据 已 经 存在 于 
数据 库 当中 。 

3. 用 户 订购 模块 
该 模块 主要 是 进行 用 户 订购 商品 操作 。 可 以 订购 多 个 商品 ， 每 个 商品 允许 输入 数量 ， 
统计 每 类 商品 的 交易 价格 ， 也 可 以 取消 没有 购买 的 订购 商品 。 但 当 用 户 确定 购买 时 ， 数 
据 会 进入 数据 库 ， 可 由 用 户 admin 查看 。 
4， 购买 物品 查看 模块 


购买 的 物品 可 以 被 管理 员 进 行 查看 ， 包 括 当日 的 订单 以 及 所 有 订单 。 订 购 系统 主要 针 
对 图 书 设计 ， 结 构 相 对 简单 ， 但 对 读者 学 习 如 何 操作 数据 库 已 经 足够 。 


18.2.3 ”在 线 订 购 系 统 数据 库 结构 


在 线 订购 系统 一 共 包 含 了 3 张 数 据 表 ， 结 构 简 单 ， 数 据 表 列表 如 表 18-1 所 示 。 


表 18-1 系统 包含 的 数据 表 
表 名 (英文 ) 
| BookInfo 
BookOrder 


图 18.1 在 线 订购 系统 流程 


说 明 
商品 (图 书 ) 信息 表 

商品 (图书) 订购 信息 表 
用 户 信息 表 


UserInfo 


在 需求 分 析 后 就 要 对 项 目 进行 初步 的 设计 ， 模 块 设计 完成 后 将 对 表 进 行 设计 。 表 结构 
的 创建 ， 可 以 帮助 代码 编写 人 员 快 速 地 掌握 业务 功能 点 ， 数 据 表 是 编码 不 可 或 缺 的 前 提 。 
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下 面 详细 地 给 出 了 这 3 张 表 的 表 结 构 。 
1. 商品 〈 图 书 ) 信息 表 (BooklInfo) 


商品 信息 表 主 要 用 于 保存 图 书 的 名 称 、 价 格 、 描 述 以 及 备注 等 信息 ， 具 体 的 表 结 构 如 
表 18-2 所 示 。 


表 18-2 商品 (图书 ) 信息 表 (Booklnfo) 


字 段 数据 类 型 说 明 

id int ID， 主 键 ， 自 增长 类 型 
bookName nvarchar(50) 商品 名 称 

bookPrice numeric(8,0) 商品 价格 

image nvarchar(300) 商品 图 片 位 置 
description nvarchar(300) 商品 简单 描述 

remark nvarchar(300) 商品 备注 


建 表 SQL 脚本 如 下 : 


CREATE TABLE [dbo].[ BookInfo] ( 
[id] [int] IDENTITY(1,1) NOT NULL, 
[bookName] [nvarchar] (50) NULL, 
[bookPrice] [decimall] (8, 0) NULL, 
[image] [nvarchar] (300) NULL, 
[description] [nvarchar] (300) NULL, 
[remark] [nvarchar] (300) NULL, 

CONSTRAINT [PK bookinfo] PRIMARY KEY CLUSTERED 

( 
[id] RASC 

) 

WITH 

(PAD INDEX = OFF, 

STATISTICS NORECOMPUTE = OFF, 

IGNORE DUP KEY = OFF, 

ALLOW ROW LOCKS = ON, 

ALLOW PAGE LOCKS = ON) 

ON [PRIMARY] 

) ON [PRIMARY] 


2. 商品 (图书 ) 订购 信息 表 (BookOrder) 
商品 订购 信息 表 主 要 用 于 保存 商品 的 订购 信息 ， 包 括 订 购 人 姓名 、 地 址 、 电 话 和 日 期 
等 信息 ， 如 表 18-3 所 示 。 


表 18-3 商品 (图书 ) 订购 信息 表 (BookOrder) 
字段 
bookOrderID ID， 主 键 ， 自 增长 类 型 
customerName nvarchar(50) 订购 人 姓名 
customerAddress nvarchar(300) 订购 人 地 址 
customerTelephone nvarchar(20) 订购 人 电话 
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续 表 
字 段 数据 类 型 说 明 


totalPrice numeric(8,0) 订购 付款 


subDate date 订购 日 期 


orderUser nvarchar(50) 登录 用 户 名 称 


myBookld 所 购 商 品 编号 
建 表 SQL 脚本 如 下 : 


CREATE TABLE [dbo] . [bookorder] ( 

bookOrderID] [int] IDENTITY (1,1) NOT NULL, 
customerName] [nvarchar] (50) NULL, 
customerAddress] [nvarchar] (300) NULL, 
customerTelephone] [nvarchar] (20) NULL, 
notic] [nvarchar] (20) NULL, 

totalPrice] [decimal] (8, 0) NULL, 

subDate] [date] NULL, 

orderUser] [nvarchar] (50) NULL, 

myBookId] [nvarchar] (300) NULL, 
CONSTRAINT [PK bookorder] PRIMARY KEY CLUSTERED 


bookOrderID] ASC 


WITH ( 
PAD INDEX = OFF, 

STATISTICS NORECOMPUTE = OFF, 
IGNORE DUP_KEY = OFF, 

ALLOW ROW LOCKS = ON, 
ALLOW_PAGE LOCKS = ON) 

ON [PRIMARY] 

) ON [PRIMARY] 


3. 用 户 信 息 表 (Userlnfo) 
用 户 信 息 表 主 要 存放 用 户 名 和 密码 等 信息 ， 表 结构 如 表 18-4 所 示 。 


表 18-4 用 户 信息 、 表 (Userlnfo) 


字段 说 明 


ID， 主 键 ， 自 增长 类 型 


userName nvarchar 用 户 名 


userld 


passWord nvarchar 密码 


bak | mvarchar | 100 | YY | 位 闪 说 明 
建 表 脚本 如 下 : 


CREATE TABLE [dbo]. [userinfo]( 
[userId] [int] IDENTITY(1,1) NOT NULL, 
[username] [nvarchar] (50) NULL, 
[password] [nvarchar] (20) NULL, 
[bak] [nvarchar] (100) NULL, 
CONSTRAINT [PK userinfo] PRIMARY KEY CLUSTERED 
( 


[userId] ASC 
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) 

WITH (PAD INDEX = OFF, 
STATISTICS NORECOMPUTE = OFF, 

IGNORE DUP KEY = OFF, 

ALLOW ROW LOCKS = ON, 

ALLOW PAGE LOCKS = ON) 

ON [PRIMARY] 

) ON [PRIMARY] 


18.3 ”在线 订购 系统 实现 


创建 数据 表 是 编码 之 前 做 的 第 一 步 。 当 完成 对 数据 库 表 的 创建 后 ， 就 可 以 开始 编码 部 
分 。 由 于 该 系统 是 B/S 结构 程序 ， 并 且 访问 SQL Server 数据 库 ， 所 以 需要 在 Tomcat 下 设 
置 对 数据 库 的 访问 。 下 面 就 如 何 访问 数据 库 ， 以 及 系统 功能 如 何 实现 做 一 个 详细 的 介绍 。 


18.3.1 JDBC Driver 的 使 用 


JDBC 的 全 称 是 Java DataBase Connectivity， 即 Java 数据 库 连 接 ， 它 是 一 套 Java 应 用 
程序 接口 ， 用 来 执行 数据 库 的 SQL 语句 ， 包 含 在 Java 类 库 中 ， 为 不 同类 型 关系 型 的 数据 
库 提供 统一 访问 。 

Java 访问 SQL Server 数据 库 时 ， 要 想 达 到 最 好 效果 ， 需 要 使 用 Microsoft SQL Server 
JDBC Driver, 该 驱动 文件 可 以 到 微软 网 站 下 载 , 下 载 地 址 是 http://www.microsoft.com/zh-cn/ 
download/details. aspx?id=21599 。 

进入 该 页 面 后 可 以 下 载 JDBC 驱动 ， 该 驱动 是 第 4 类 型 的 驱动 ， 需 要 运行 在 Java 5 以 
及 更 高 的 版 本 中 ， 读 者 可 以 参考 图 18.2。 


下 载 SQL Server JDBC Driver 3.0， 这 吓 一 个 Type 4 JDBC 驱动 程序 ， 它 通过 Java 
Platform, Enterprise Edition 5 及 种 高 版 本 中 可 用 的 标准 ]DBC 应 用 程序 纺 程 接口 (API) 


提 共 妆 据 库 连接 。 
快速 详细 信息 
[2 30.1301101 发 和 日 其 - 2010/a/20 
ER 六 [zag | 
此 下 载 中 包 合 的 文件 
此 名 中 和 入 反 与 活用 于 比 下 和 和 文件 区 局 。 下 二 呈 到 并 二 . 
立 填 扣 大 小 
:am 


图 18.2 JDBC 驱动 下载 
全 提示 : 这 里 选择 大 小 为 “3.7MB” 的 下 载 即 可 。 
首先 保证 计算 机 有 Java 运行 环境 (JRE)， 如 果 没 有 ， 建 议 读者 安装 JDK 1.6， 因 为 本 
章 的 系统 运行 在 JDK 1.6 和 Tomcat 6 环境 下 。 接 下 来 ， 读 者 需要 双击 运行 
sqljdbe_3.0.1301.101_chs.exe 文件 〈 利 用 压缩 文 解压 缩 也 可 以 )。 在 安装 目录 或 解压 目录 中 
的 sqljjdbc 3.0\chs 下 可 以 看 到 有 两 个 以 .jar 为 后 绥 的 文件 ， 这 两 个 文件 是 Microsoft SQL 
Server JDBC Driver 提供 的 类 库 文件 ， 它 们 分 别 是 sqljdbc.jar 和 sqljdbc4.jar， 这 两 个 文件 说 


.372 。 


第 18 章 JSP 在 线 订购 系统 


明 如 下 : 
口 sqljdbcjar: 它 提供 对 JDBC 3.0 的 支持 ， 要 求 Java 运行 环境 (JRE) 为 5.0， 其 他 
版 本 的 运行 环境 会 提示 错误 。 
口 sqljdbc4.jar: 它 提供 对 JDBC 4.0 的 支持 ， 是 sqljdbc.jar 的 超 集 。 要 求 在 版 本 为 6.0 
或 更 高 的 Java 运行 环境 (JRE) 下 使 用 ， 其 他 版 本 上 会 引发 异常 。 

这 里 读者 需要 把 sqljdbc4.jar 文件 复制 到 Tomcat 6 下 的 lib 文件 夹 中 , 以 加 载 驱动 程序 ， 
也 可 以 复制 到 项 目 lib 包 中 。 

18.3.2 ”连接 数据 库 

Java 连接 数据 库 大 体 上 可 以 利用 两 种 方式 : 一 种 是 Java Database Connectivity (JDBC) 
连接 数据 库 ， 另 一 种 则 是 利用 Java Naming and Directory Interface (JNDI) 来 操作 数据 库 。 
其 中 JNDI 中 连接 数据 库 需 要 容器 的 支持 ， 而 JDBC 则 不 需要 。JDBC 操作 简单 ， 在 小 程序 
中 应 用 没有 问题 ， 也 适合 初学 者 。 

JDBC 全 称 是 Java DataBase Connectivity， 它 是 一 套 Java 应 用 程序 接口 ， 为 不 同 数据 
库 提 供 统 一 访问 。 另外 ， 由 于 Java 程序 要 运行 在 虚拟 机 中 ， 因 此 由 Java 编写 的 程序 可 以 
运行 到 任何 支持 Java 平台 的 环境 下 ， 这 样 ，Java 就 做 到 了 “一 次 编写 ， 随 处 运行 ”。 

JNDI 全 称 是 Java at and Directory Interface， 被 用 于 执行 名 字 和 目录 服务 。 它 提 
供 了 一 致 的 模型 来 存 取 和 操作 企业 级 的 资源 , JINDI 目录 机 构 中 的 每 个 节点 被 称 为 Context。 
每 一 个 JNDI 名 字 都 是 相对 于 Context 的 ， 同 时 通过 目录 树 来 定位 它 所 需要 的 资源 。 在 利用 
JNDI 连接 数据 库 时 ， 被 创建 的 DataSource 会 放 入 到 JNDI 范围 内 ， 假 如 服务 器 是 Tomcat, 
那么 DataSource 会 放 入 Tomcat 内 ， 然 后 直接 从 该 服务 器 中 获取 就 可 以 使 用 了 。 
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下 面 给 出 了 DBConnection 类 的 代码 , 该 类 实现 了 利用 JDBC 获取 数据 库 连 接 ,位 于 init 
包 内 。 


01 package app.init; 

02 

03 // 引 入 资源 包 

04 import java.sql.Connection; 

05 import java.sql.DriverManager; 
06 :import java.sql.SQLException; 


07 

08 /2 

09 “* 连接 类 

10 * @author Administrator 
二 区 


12 public class DBConnection { 

13 ”// 声明 变量 并 赋值 

14 private static String drivers = "com.microsoft.sqlserver. 
jdbc.SQLServerDriver"; 

15 private static String url = "jdbc:sqlserver://localhost:1433; 
DatabaseName=cart18"; 

16 private static String user = "sa"; 

17 private static String password = "root"; 

18 

ho 


es 
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20 * 获取 数据 库 连 接 ， 返 回 Connection 对 象 


2 

22 * @return 

231 *X 

24 public static Connection GetConnection() { 

之 5 Connection conn = null; 

26 Etrey 

pr //Class .forName () 方法 创建 驱动 程序 实例 ， 同 时 调用 DriverManager 
对 其 注册 

28 Class.forName (drivers) .newInstance () ; 

29 } catch (InstantiationException e) { 

30 e.printStackTrace () 

34 } catch (IllegalAccessException e) { 

3 区 e.printStackTrace () 

33 } catch (ClassNotFoundException e) { 

34 e.printStackTrace (); 

ED } 

36 

EM ey 

38 // 通过 DriverManager 获取 数据 库 连 接 

39 conn = DriverManager.getConnection (url, user, password); 

40 } catch (SQLException e) { 

41 e.printStackTrace () 

42 } 

43 

44 return conn; 

45 } 

46 

全 

48 ”* 关闭 连接 

9 

50 * @param conn 

to 

52 public static void close (Connection conn) { 

3 人 3 try { 

54 // 要 先 判 断 是 否 为 NULL 才能 判断 是 否 关 闭 

号 3 if (conn != null && !conn.isClosed() 

56 conn.close (); 

sl } catch (SQLException e) { 

58 e.printStackTrace (); 

59 

60 J} 

61 

其 中 : 

口 第 1 行 ， 表 示 该 类 所 在 的 包 。 

口 第 4 一 6 行 ， 表 示 该 类 需要 引用 的 其 他 类 。 

口 第 12 行 ， 表 示 创 建 该 类 ， 名 称 为 DBConnection， 是 一 个 公共 类 。 

口 第 14 一 17 行 , 创建 入 有 变量 ， 并 初始 化 连接 数据 库 参 数 ， 其 中 drivers 是 数据 库 驱 


动 串 ，url 是 数据 库 连 接 串 。 
第 24 行 ， 表 示 创 建 静态 方法 GetConnection， 该 方法 返回 一 个 Connection 对 象 。 
第 26~35 行 ， 是 一 个 捕捉 异常 语句 块 ， 该 语句 块 里 面 的 代码 有 可 能 发 生 异常 ， 是 
强制 捕捉， 提高 了 代码 的 健壮 性 。 

口 第 28 行 ， 创 建 驱动 程序 实例 ， 同 时 利用 DriverManager 对 其 注册 。 

口 第 39 行 ， 利 用 提供 的 参数 获取 当前 数据 库 连 接 。 
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口 第 和 4 行 ， 返回 数据 库 连 接 对 象 。 

口 第 52 行 ， 创建 静态 方法 colse， 该 方法 含有 一 个 参数 ， 参 数 是 Connection 对 象 ， 
它 的 作用 是 把 传递 进去 的 连接 对 象 进行 关闭 操作 。 

口 第 55 行 ， 判 断 方法 传 进 对 象 是 否 为 空 ， 是 否 被 关闭 。 

口 第 56 行 ， 假 如 参数 不 是 空 ， 没 有 被 关闭 ， 那 么 执行 关闭 操作 。 

当 用 户 需要 操作 数据 库 时 ， 直 接 调 用 该 类 中 的 方法 获取 连接 即 可 ， 非 常 方便 ， 该 类 被 

的 地 方 在 BookOrderDAO.java 中 。 相 关 调 用 代码 如 下 

OA 

02 * 查看 所 有 订购 

93 

04 * @return 

05 * @throws SQLException 

(Ve 

07 public List<BookOrder> getBookOrderListAll() throws SQLException { 

08 List<BookOrder> list = new ArrayList<BookOrder>(); 

09 Connection con = DBConnection.GetConnection(); 

10 PreparedStatement pstmt = null; 

11 String sql = "SELECT * FROM BOOKORDER WHERE ORDERUSER IS NOT NULL 

人 " +"ORDER BY SUBDATE"; 

3 BookOrder bookorder = null; 

14 try 

LS pstmt = con.prepareStatement (sql); 

16 // 执行 

下 了 ResultSet rs = pstmt.executeQuery(); 

18 while (rs.next()) { // 取 结果 集中 的 结果 

19 bookorder = new BookOrder (); 

20 bookorder.setCustomerName (rs.getString (2)); 

2 bookorder.setCustomerAddress (rs.getString (3)); 

22 bookorder.setCustomerTelephone (rs.getString(4)); 

3 bookorder.setNotic(rs.getString(5)); 

24 bookorder .setTotalPrice (rs.getDouble (6)); 

2 bookorder.setSubdate (String.valueOf (rs.getDate (7))); 

26 bookorder.setSessuser (rs.getString (8)); 

27 bookorder. setMyBookId (rs.getString (9)); 

28 

29 list.add (bookorder); // 向 1ist 对 象 中 装载 Bookorder 对 象 

30 } 

31 rs.close(); 

32 } finally { 

33 pstmt.close(); 

34 con.close(); 

35 : 

36 return list; 

Sl 

其 中 : 

口 第 1 一 6 行 ， 该 方法 的 注释 。 

口 第 7 行 ,表示 创建 方法 getBookOrderListAll， 不 带 参数 ， 返 回 list 对 象 ， 主 动 抛 出 
异常 。 

口 第 8 行 ， 声 明 变 量 list 并 实例 化 ， 该 实例 中 运行 存储 BookOrder 对 象 。 

口 第 9 行 ， 获 取 Connection， 调 用 DBConnection 类 。 

口 第 10 行 ， 创 建 PreparedStatement 对 象 。 
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第 11 行 ， 声 明 String 类 型 变量 ， 并 初始 化 ， 变 量 内 容 是 SQL 语句 。 
第 13 行 ， 声 明 BookOrder 对 象 。 
第 15 行 ， 利 用 SQL 语句 实例 化 PreparedStatement 对 象 。 
第 17 行 ， 执 行 查询 操作 ， 并 返回 结果 集 。 
第 18 行 ， 输 出 结果 集中 的 数据 ， 并 把 数据 存 入 BookOrder 对 象 中 。 
第 29 行 ， 向 list 对 象 中 装载 BookOrder 对 象 。 
第 31 行 ， 关 闭 结果 集 。 
第 33 行 ， 关 闭 语句 集 。 
第 34 行 ， 关 闭 连 接 。 
关闭 结果 集 、 语 句 集 、 连 接 ， 这 个 顺序 不 能 变 ， 不 然 容 易 内 存 泄漏 出 错 。 
第 36 行 ， 返 回 list 对 象 。 
在 以 上 代码 中 ， 利 用 前 面 介绍 的 JDBC 连接 了 数据 库 ， 并 从 数据 库 查询 数据 。 具 体 执 
行 效 果 如 图 18.3 所 示 。 


DoOOOOOOOOO DO 


口 、 
商 品 订 购 系 统 
用 户 今天 订购 商品 : 查看 所 有 
蚁 号 用 户 单价 物品 编号 订单 附 言 BM 
1 tt 0 因 aoe-t2-at 
返回 


图 18.3 利用 JDBC 查询 数据 库 数据 


JDBC 的 使 用 适合 初学 者 ， 但 在 实际 的 Java Web 项 目 开 发 中 很 少 使 用 JDBC 来 操作 数 
据 库 ， 因 为 在 标准 的 JDBC 接口 中 ， 并 没有 提供 资源 的 管理 方法 ， 在 访问 量 大 的 系统 中 显 
得 效率 较 低 ， 要 想 提高 连接 数据 库 的 效率 ， 有 必要 使 用 连接 池 。JDBC 下 使 用 连接 池 则 需 
要 开发 者 自己 编写 相关 代码 。 而 JNDI 则 非常 方便 地 解决 了 这 个 问题 ， 简 单 地 配置 脚本 ， 
就 可 以 完成 对 数据 库 的 操作 。 

2. 利用 JNDI 连接 数据 库 


下 面 给 出 了 利用 JNDI 获取 连接 的 操作 步骤 。 

(1) 在 Eclipse 中 设置 项 目 Cart18 为 Tomcat 项 目 ， 如 图 18.4 所 示 。 

当 完 成 该 步骤 ， 那 么 在 tomcat 目录 conf\Catalina\localhost 下 会 生成 一 个 名 为 cart18 的 
xml 配置 文件 ， 该 配置 文件 需要 修改 。 

(2) 修改 cart18.xml 文件 ， 修 改 后 内 容 如 下 : 


<Context path="/cartl18" reloadable="true" 
docBase="D: \NUTZ\Cart18\WebRoot" workDir="D:\NUTZ\Cart1l8\work" > 
<Resource 
name=" jdbc/cart18" 
type="javax.sql.DataSource" 
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Tomcat 国生 


Gaeral |netosaur Chessvath | zxoort to ma settines| 


Is Tomcat Project 


EC 

VS Can update context definition (server xmal or context file) 
IF Nark this context as reloadsble (relosdsble="true") 

厂 Redirect context logger to Eclipse console 


Extrs information 


型 


Subdirectory to set as web spplication root (optional) csrtls 


人 
图 18.4 标记 项 目 为 Tomcat 项 目 


driverClassName="com.microsoft.sqlserver.jdbc.SQLServerDriver" 
maxIdle="4" 
maxWait="3000" 
url="jdbc:sqlserver://localhost:1433;DatabaseName= cart18" 
username="sa" 
password="root" 
maxActive="3" 

> 

</Context> 


刁 


path: Web 应 用 的 上 下 文 路 径 。 
reloadable: 当 服 务 器 开启 时 ， 修 改 程序 中 的 代码 后 ， 服 务 器 会 自动 重新 加 载 。 
docBase: Web 应 用 程序 所 在 的 目录 。 
workDir: 编译 后 文件 所 在 目录 。 
name: DataSource 的 名 称 ， 该 名 称 要 和 web.xml 文件 中 的 名 称 相同 。 
type: 数据 源 对 应 的 java 类 型 ， 使 用 javax.sql.DataSource。 
driverClassName: JDBC 驱动 程序 名 。 
maxIdle: 绥 冲 池 人 允许 的 处 于 空闲 状态 的 最 大 数目 的 数据 库 连 接 ，0 表示 不 做 限制 。 
maxWait: 缓冲 池 中 的 数据 库 连接 处 于 空闲 状态 的 最 长 时 间 , 0 表示 不 做 时 间 限 制 。 
url: JDBC 连接 数据 库 的 连接 地 址 ，localhost 表示 本 地 服务 器 。 
username: 数据 库 用 户 名 。 
password: 数据 库 用 户 名 对 应 的 密码 。 
maxActive: 缓冲 池 中 活动 状态 的 数据 库 连 接 的 最 大 数 。 
(3) 修改 web.xml 文件 内 容 ， 该 文件 在 Cart18 项 目下 的 WEB-INF 目录 中 ， 在 其 中 的 
<web-app></web-app> 之 间 添 加 如 下 脚本 : 
<resource-ref> 
<description>DB Connection</description> 
<res-ref-name> jdbc/cart18</res-ref-name> 
<res-type>javax.sql.DataSource</res-type> 


<res-auth>Container</res-auth> 
</resource-ref> 
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其 中 : 

口 description: 简单 描述 。 

口 res-refname: jdbc/cart18 要 和 第 2 步 中 name 元 素 值 相同 。 

以 上 为 JNDI 的 配置 文件 ， 要 想 连 接 数 据 库 ， 还 需要 创建 类 ， 以 便 获取 数据 库 连 接 。 

(4) 编写 代码 ， 获 取 数 据 库 连接 。 这 里 类 名 为 FactoryConn， 获 取 数 据 库 连 接 的 具体 代 
码 如 下 : 

01 package app.dao; 

02 

03 import java.sql.Connection; 

04 import java.sql.SQLException; 

05 

06 import javax.naming.InitialContext; 


07 import javax.naming.NamingException; 
08 import javax.sql.DataSource; 


09 

9 /大 

11 * JNDI 

12 * @author Administrator 

3 

14 public class FactoryConn { 

Em 

16 * 获得 连接 

L7 * @return 

18 A 

19 public static Connection getConnection() { 

20 Connection conn = null; // 声 明 连接 对 象 

21 // 获 取 连 接 ， 异 常 捕捉 块 

22 Ey 

23 // 初 始 化 上 下 文 ， 获 取 数 据 源 

24 InitialContext ctx = new InitialContext() 

25 DataSource ds = (DataSource) ctx.lookup("java:comp/env/ 
ybc/cart1on)y 

26 try 渡 

27 conn=ds.getConnection(); // 从 数据 源 获取 连接 

28 } catch (SOLException e) { 

29 // 出 现 异常 

30 e.printStackTrace (); 

Ee } 

32 } catch (NamingException ex) { 

33 ex.printStackTrace (); 

34 水 

35 

36 return conn; // 返 回 连接 对 象 

= 7 

38 1} 

其 中 


口 第 1 行 ， 该 类 包 名 ， 即 app.dao。 

口 第 14 行 ， 创 建 公 共 类 ， 类 名 为 FactoryConn 。 

口 第 19 行 ， 创 建 静态 方法 getConnection， 返 回 Connection 实例 。 
口 第 20 行 ， 声 明 Connection 连接 对 象 ， 名 为 conn， 初 始 值 为 null。 
口 第 22 行 ， 异 常 捕 提 块 。 

口 第 24 行 ， 创 建 初始 上 下 文 对 象 ， 对 象 名 为 ctx。 


第 25 行 ， 
第 27 行 ， 
36 行 ， 
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获取 数据 源 ， 即 DataSource 对 象 。 
从 数据 源 获取 连接 。 
返回 连接 对 象 。 


le 只 需要 调用 getConnection() 方 法 ， 就 可 以 获取 数据 库 
后 进行 相关 操作 。 相 关 调 用 代码 如 下 : 


package app.dao; 


import 
import 
import 
import 
import 
import 


import 


/** 


java.sql.Connection; 
java.sql.PreparedStatement; 
java.sql.ResultSet; 
java.sql.SQLException; 
java.util.ArrayList; 
java.util.List; 


app .bean.Book; 


* @author Administrator 


大 

i 
public 
/** 


class BookDAO { 


* 查询 所 有 数据 
* 返回 LIST 对 象 


* @return 


wf 
public 


List<Book> getAllBooks() throws SQLException { 
List<Book> books = new ArrayList<Book>(); // 创 建 List 对 象 
Connection con = FactoryConn.getConnection(); // 获 取 连 接 
PreparedStatement pstmt = null; 
String sql = "SELECT * FROM BOOKINFO"; // 编 写 SQL 语句 
try 
pstmt = con.prepareStatement (sql); 
// 执行 
ResultSet rs = pstmt.executeQuery(); 
while (rs.next()) { // 取 得 结果 集中 的 值 
Book book = new Book(); 
book.setId (rs.getLong("id") ) 
book .setBookName (zs .getString("bookName") ) 7 
book .setRemark (zs .getString("remark") ) 7 
book.setBookPrice(rs.getDouble ("bookPrice")); 
book .setBkImage (rs.getString ("image")); 
book.setDescription(rs.getString ("description")); 
// 添加 到 books 
books .add (book) 
} 
rs.close(); // 关 闭 结果 集 


} finally { 


上 


pstmt.close(); 
con.close(); 


return books; 


7 
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Se 

区 * @param bookid 

54 * @return 

5 * @throws SQLException 

6 

57 public Book getDetail (Long bookid) throws SQLException { 

58 Book book = new Book(); // 创 建 Book 对 象 
59 Connection con = FactoryConn.getConnection(); // 获 取 连 接 
60 PreparedStatement pstmt = null; 

61 String sql = "SELECT * FROM BOOKINFO WHERE ID=?"; // 编 写 SQL 语句 
62 try + 

63 pstmt = con.prepareStatement (sql); 

64 // 占 位 符 

65 pstmt.setLong(1, bookid); 

66 ResultSet rs = pstmt.executeQuery() ;// 将 查询 结果 保存 到 结果 集中 
67 while (rs.next()) { / /查询 结果 集中 的 值 

68 book.setId (rs.getLong("id") ) 

69 book.setBookName (rs.getString ("bookName")); 

70 book.setRemark (rs.getString ("remark")); 

71 book.setBookPrice(rs.getDouble ("bookPrice")); 

了 2 book.setBkImage (rs.getString ("image")); 

3 book.setDescription(rs.getString("description") ) 

74 } 

75 } finally { 

76 pstmt.close(); 

法 坟 con.close(); 

78 } 

79 return book; 

80 

81 } 

e200 

其 中 : 

口 第 16 行 ， 创建 类 ， 类 名 为 BookDAO。 

口 第 22 行 ， 创建 方法 ， 获 取 List 对 象 ， 该 对 象 内 存放 的 是 Book 对 象 。 

口 第 23 行 ， 创 建 List 对 象 Books， 并 初始 化 。 

口 第 24 行 ， 利 用 FactoryConn 类 中 的 getConnection() 方 法 获取 数据 库 连 接 ， 这 里 实 


际 上 是 获取 了 缓冲 池 中 的 连接 。 

第 25 行 ， 创 建 PreparedStatement 对 象 ， 赋 值 null。 

第 26 行 ， 创 建 字符 串 ， 赋 值 为 SQL 语句 。 

第 27 行 ， 异 常 捕捉 块 开 始 。 

第 28 行 ， 利 用 连接 对 象 初始 化 PreparedStatement 对 象 。 
第 30 行 ， 查 询 ， 获 取 结 果 集 。 

第 31 一 38 行 ， 过 历 结果 集 ， 提 取 数 据 放 入 Book 对 和 象 中 。 
第 40 行 ， 将 Book 对 象 放 入 List 对 象 中 。 


OOOOOOO DO 


方法 即 可 。 
当 获 取 数 据 库 连接 的 问题 解决 之 后 ， 就 可 以 实现 其 他 具体 的 功能 点 ， 例 如 月 


查看 商品 、 订 购 商 品 以 及 确认 购买 等 。 


“380。 


第 50 一 81 行 ， 根 据 参数 获取 明细 方法 。 这 里 不 详细 介绍 ， 读 者 参考 getAllBooks 


旧 户 登录 、 
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18.3.3 ”实现 登录 功能 


登录 操作 通常 是 商业 系统 中 用 户 首要 进入 的 页 面 , 本 系统 中 , 读者 可 以 在 Eclipse 中 启 
动 Tomcat 服务 器 ,然后 利用 浏览 器 进入 登录 页 面 ， 登 录 页 面 地 址 为 http://localhost:8088/ 
cartl8/login.jsp。 

如 无 其 他 问题 ， 浏 览 器 将 进入 图 18.5 所 示 的 页 面 ， 该 页 面 是 展示 部 分 ， 由 JSP 完成 。 
除了 展示 页 面 外 ， 后 台 还 有 业务 逻辑 以 及 数据 持久 化 部 分 。 


商品 订购 系统 
| 


用 户 各 : 


图 18.5 系统 登录 页 面 


在 图 18.5 所 示 页 面 中 输入 用 户 名 密码 ， 管 理 员 可 以 用 admin， 对 应 密码 为 123， 测 试 
用 户 为 test， 对 应 密码 为 test。 有 关 用 户 已 经 预先 存 入 数据 库 当中 ,读者 可 以 直接 使 用 。 在 
登录 页 面 用 户 名 输入 test， 密 码 同样 输入 test， 单 击 “ 登 录 ” 按 钮 ， 会 转 入 商品 列表 页 面 。 

登录 操作 主要 分 为 以 下 3 个 步骤 。 

(1) 在 UserCmd 文件 中 获取 JSP 页 面 传递 过 来 的 数据 ， 主 要 包括 用 户 名 和 密码 。 

(2) 传递 到 后 台 的 用 户 名 和 密码 将 作为 查询 条 件 ， 对 数据 库 进 行 查询 。 

(3) 获取 查询 结果 ， 并 判断 该 查询 结果 是 否 有 效 。 当 有 效 时 ， 则 表示 用 户 名 和 密码 相 
互 对 应 ， 可 以 登录 ; 反之 ， 则 不 允许 登录 。 

该 程序 中 ， 实 现 登录 功能 的 主要 文件 有 login.jsp、UserCmd.java 和 UserDAO.java。 下 
面 分 别 对 两 个 类 文件 进行 详细 说 明 。 


1. UserCmd 类 的 实现 


它 的 作用 是 获取 前 台 页 面 , 即 JSP 中 传递 的 数据 , 并 根据 实际 需求 进行 业务 风 辑 操作 ， 
其 相关 代码 如 下 : 


01 public void doGet (HttpServletRequest request, HttpServletResponse 


response) 
02 throws ServletException, IOException { 
03 String userName = request.getParameter ("loginName"); 
04 String password = request.getParameter ("password"); 
05 String sessuser = null; 
06 // 交易 价格 
07 double tfp = 0.0; 
08 Er 
09 // 根据 用 户 名 和 密码 获取 数据 


= 


第 5 篇， 数据 库 的 应 用 


10 User user = new DoUser () .getUser (userName, password); 

h if (user == null) { // 如 果 数 据 是 空 

了 2 request.setAttribute ("message"，" 登 录 名 或 密码 错误 ! ") ; 

13 request.setAttribute ("userName", userName); 

14 request.getRequestDispatcher ("/login.jsp"). 
forward (request, 

5 response); 

16 return; 

1 } 

18 Map<Book, Integer> ptbooklist = (Map) request.getSession() 

19 .getAttribute ("SESS BOOK"); 

20 

2 if (ptbooklist != null) { 

有 2 公 request.getSession() .removeRttribute("SESS_BOOK") ， 

23 } 

24 // 如 果 用 户 名 和 密码 有 对 应 的 数据 

25 // 查看 SESSION 中 SESS_USER 键 是 否 有 对 应 数据 

26 sessuser = (String) request.getSession() .getAttribute 

("SESS_USER"); 

27 // 交易 价格 放 入 SESSION 

28 request.getSession() .setAttribute("SESS TFP", tfp); 

29 

30 // 如 果 sessuser 不 为 空 , 移 除 SESS_USER 对 应 数据 

3 if (sessuser != null) { 

32 request.getSession() .removeAttribute ("SESS USER"); 

33 } 

34 // 重新 设置 SESSION 中 SESS_USER 键 对 应 的 用 户 名 

35 request.getSession().setAttribute ("SESS USER", userName); 

36 

3 response.sendRedirect (request.getContextPath() + 

"/booklist"); 

38 } catch (SQLException e) { 

39 e.printSstackTrace (); 

40 } 

41 } 

42 


43 Q@Override 
44 public void doPost (HttpServletRequest request, HttpServletResponse 


response) 
45 throws ServletException, IOException { 
46 
47 this.doGet (request, response); 
48 } 
其 中 : 


口 第 1 行 , 创建 类 ， 该 类 实际 上 是 个 Servlet， 重 写 方法 doGet。 

口 第 3 行 ， 声明 变量 userName， 从 页 面 获取 用 户 名 参数 ， 并 赋值 该 变量 。 

口 第 4 行 ， 声 明 变量 password， 从 页 面 获 取 密 码 参数 ， 并 赋值 该 变量 。 

口 第 5 行 ， 声明 变量 sessuser， 其 初始 化 为 null， 该 变量 将 来 存放 SESSION 变量 。 

口 第 7 行 ， 声 明 变 量 tp， 交 易 价 格 。 

口 第 8 行 ， 异 常 捕捉 块 开 始 ， 该 语句 块 内 的 异常 将 被 捕捉 。 

口 第 10 行 ， 声 明 User 对 象 ， 并 调用 DoUser 中 的 方法 返回 User 实例 。 

口 第 11 行 ,判断 返回 user 是 否 为 null， 假 如 为 null， 将 返回 操作 ， 说 明 用户 名 和 密 
码 不 匹配 。 


=“ 382 


口 


口 


2 


传递 到 


UserCmd 当中 。UserDAO 类 部 分 代码 如 下 : 
01 public class UserDRO { 
0 
03 ”* 根据 用 户 登录 名 和 登录 密码 得 到 用 户 
04 * 
05 * @param userName 
06 * @param password 
07 * @return 
08 * @throws SQLException 
SR 
10 public User getUserNameAndPwd (String userName, String password) 
i throws SQLException { 
2 User user = null; 
LE3 Connection con = FactoryConn.getConnection(); 
14 PreparedStatement pstmt = null; 
15 String sql = "SELECT USERID,USERNAME,PASSWORD " 
16 + "FROM USERINFO WHERE USERNAME = ? AND PASSWORD = ?"7 
17 try { 
18 
19 pstmt = con.prepareStatement (sql); 
20 // 占 位 赋值 
2 pstmt.setString(1, userName); 
2 pstmt.setString(2, password); 
23 
24 ResultSet rs = Pstmt .executeoOuery() 
25 while (rs.next()) { // 遍历 结果 集 , 获取 数据 
26 user = new User(); 
27 user.setUserID(rs.getLong ("userid")); 
28 user.setUserName (rs.getString ("username")); 
29 user.setPassword(rs.getString ("password")); 
30 1 
3 rs.close(); 
32 } finally { 
3 pstmt.close(); 
34 con.close(); 
35 } 
36 return user; 
37 
38 } 
39 
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第 12 一 13 行 ， 表示 利用 request 对 象 向 请 求 中 赋值 。 
第 14 行 ， 表 示 跳 转 页 面 。 


第 18 行 ， 表 示 从 SESSION 中 获取 名 为 SESS_BOOK 对 应 的 数据 ， 它 对 应 订购 


列表 。 


第 21 一 23 行 , 表示 判断 订购 列表 是 否 为 空 , 假如 不 为 空 ， 则 需要 清除 该 SESSION 
因为 新 用 户 需 要 新 的 值 。 下 面 第 26 行 实际 在 做 着 同样 的 操作 ， 不 过 是 针对 用 


值 ， 


户 名 的 。 
第 37 行 ， 跳 转 命令 ， 调 用 booklist， 这 是 一 个 Servlet。 
第 44 行 ， 重 写 doPost 方法 ， 这 里 调用 了 doGet 方法 。 


UserDAO 类 的 实现 


该 类 主要 获取 数据 库 连 接 , 并 根据 预定 的 SQL 语句 执行 查询 , 把 从 数据 库 获 取 的 数据 


*383，* 


当 
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其 中 : 


第 1 行 ， 创 建 UserDAO 类 ， 该 类 访问 数据 库 。 
第 10 行 ， 创 建 getUserNameAndPwd 方法 ， 并 主动 抛 出 异常 。 
第 12 行 ， 声 明 User 对 象 ， 赋 值 为 null。 
第 13 行 ， 获 取 数 据 库 连接 ， 初 始 化 con 变量 。 
第 14 行 ， 声 明 PreparedStatement 对 象 ， 并 初始 化 为 null。PreparedStatement 对 象 
可 以 多 次 而 且 高 效 地 执行 SQL 语句 。 
第 15 一 16 行 ， 把 SQL 语句 赋值 给 String 变量 中 。 
第 17 行 ， 异 常 捕捉 块 开 始 。 
第 19 行 ， 为 pstmt 赋值 。 
第 21 一 22 行 ， 为 SQL 语句 中 的 占 位 符 赋值 (问号 表示 占 位 符 )。 
第 24 行 ， 得 到 结果 集 。 
第 25 一 30 行 ， 使 用 while 语句 遍历 ResultSet 对 象 中 的 数据 ， 并 把 结果 存 入 user 
实例 中 。 
第 31 行 ， 关 闭 结 果 集 。 
第 33 行 ， 关 闭 当前 数据 库 连 接 。 
登录 成 功 后 ， 会 进入 商品 列表 页 面 ， 如 图 18.6 所 示 。 


商品 订购 系统 


的 :1 Ln 
甲 帮 本 红楼梦 

a 价格 : 50.0 元 

的 :5 EE 


图 18.6 商品 列表 页 面 


18.3.4 实现 商品 列表 功能 


商品 列表 页 面 将 列 出 所 有 已 经 在 数据 库存 在 的 商品 ， 该 功能 本 质 上 是 一 个 数据 查询 ， 
实现 相对 简单 ， 有 具体 查询 效果 读者 可 以 参考 图 18.6， 后 台 实 现 主要 有 BookListCmd 和 
BookDAO 两 个 类 。 


‘es 


该 类 只 需要 调用 访问 数据 库 的 代码 即 可 ， 然 后 把 返回 的 数据 赋值 到 JSP 页 面 上 ， 并 进 


BookListCmd 类 的 实现 


行 页 面 跳 转 操作 。BookListCmd 类 部 分 代码 如 下 : 
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public void doGet (HttpServletRequest request, HttpServletResponse 
response) 
throws ServletException, IOException { 
TY 


List<Book> books = new DoBook () .1istBook() ; // 创 建 List 对 象 books 
request.setAttribute ("books"，books); // 把 数据 保存 到 request 中 


request.getRequestDispatcher ("/list.jsp") // 跳 转 页 面 
.forward (request， response); 


} catch (SQLException e) { 
e.printStackTrace (); 


} 


其 中 : 


第 1 行 ， 重 写 Servlet 方法 doGet。 

第 4 行 ， 声 明 变 量 books， 是 List 类 型 对 象 ， 获 取 数 据 库 查询 数据 。 
第 5 行 ， 把 数据 保存 到 request 中 。 

第 6 行 ， 跳 转 到 页 面 ， 页 面 将 获取 查询 数据 。 

BookDAO 类 的 实现 

该 类 用 于 访问 数据 库 操作 ,并 把 从 数据 库 获 取 的 数据 传递 给 调用 它 的 BookListCmd 类 。 


BookDAO 类 相关 代码 如 下 : 


01 
02 


03 
04 
05 
06 
07 
08 
09 
10 
二 击 
12 
13 
14 
15 
16 
ED 
18 
19 
20 
1 
22 
23 
24 
25 
26 
7 


public List<Book> getAllBooks() throws SQLException { 
List<Book> books = new ArrayList<Book>(); // 创 建 List 对 象 
Connection con = FactoryConn.getConnection(); // 获 取 连 接 
PreparedStatement pstmt = null; 


String sql = "SELECT * FROM BOOKINFO"; // 编 写 SQL 语句 
wy 

pstmt = con.prepareStatement (sql); 

// 执行 

ResultSet rs = pstmt.executeQuery(); 

while (rs.next()) { // 遍 历 结果 集 


Book book = new Book(); 
book.setId(rs.getLong ("id")); 
book.setBookName (rs.getString ("bookName")); 
book.setRemark (rs.getString ("remark")); 
book.setBookPrice(rs.getDouble ("bookPrice")); 
book.setBkImage (rs.getString ("image")); 
book.setDescription(rs.getString ("description")); 
// 添加 到 books 
books .add (book); 
rs.close(); 
} finally { 
pstmt.close(); 
con.close(); 
} 
return books; 


其 中 : 


ws 


口 
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第 1 行 , 创建 getAllBooks 方法 , 该 方法 根据 条 件 查 询 数据 库 , 获取 所 有 商品 列表 ， 


返回 数据 列表 。 


间 


人 OOOOOOODDO 


18.3.5 
在 商 


第 2 行 ， 创建 List 对 象 books， 并 被 初始 化 。 

第 3 行 ， 从 FactoryConn 中 获取 数据 库 连 接 。 

第 4 行 ， 创 建 PreparedStatement 对 象 ， 初 始 值 是 null。 

第 5 行 ， 创 建 字 符 串 对 象 ， 存 入 SQL 语句 。 

第 7 行 ， 实 例 化 pstmt 对 象 。 

第 9 行 ， 获 取 结 果 集 。 

第 10 一 19 行 ， 遍 历 ResultSet 对 象 中 的 数据 ， 并 把 结果 存 入 List 对 象 中 。 
第 26 行 ， 返 回 结果 。 


品 列表 页 面 可 以 订购 自己 需要 的 商品 ， 并 提交 订单 ， 完 成 订购 。 
实现 商品 订购 功能 
品 列表 页 面 ， 列 出 了 已 经 存在 数据 库 中 的 所 有 商品 ， 登 录用 户 可 以 根据 自己 的 需 


求 订购 已 经 存在 的 商品 ， 订 购 商 品 的 步骤 如 下 : 


Cl 


在 图 18.6 所 示 页 面 中 单 击 “ 订 购 ” 链 接 ， 会 弹出 一 个 简单 的 订购 数量 对 话 框 ( 如 


果 IE 没 弹 出 对 话 框 ， 也 会 有 提示 ， 读 者 只 需要 允许 执行 窗口 脚本 即 可 )， 在 该 对 话 框 中 输 
入 具体 订购 的 数量 ， 如 图 18.7 所 示 。 


商品 订购 系统 


钨 号: 1 [SEE a 广 叶 -4 [5 误 习 


甲 挟 本 红楼 莽 本 如 世 
价格 : 60 0 元 I * 
Te 


图 18.7 输入 订购 商品 


(2) 当 订 购 商 品 被 确定 后 允许 多 订 )， 会 进入 订购 列表 中 ， 登 录用 户 可 以 单 击 页 面 
中 的 “确认 购买 ”按钮 ， 如 图 18.8 所 示 ， 进 入 订单 提交 页 面 。 
商品 订购 系统 
蝙 号 单价 和 和 本 数量 分 类 价格 


“386。 


图 18.8 订购 列表 
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(3) 在 订单 提交 页 面 需要 填写 个 人 信息 ， 如 图 18.9 所 示 。 个 人 信息 确认 无 误 后 ， 用 户 


单 击 “ 提 交 ” 按 钮 ， 确 认购 买 。 


确认 个 人 信息 


订购 人 ， 网 
订购 人 地 址 ，。 网 式 地 证 
联系 电话 ， [3500000000 
付费 金 新 ， 62. 0 元 
网 区 条 再 


订单 说 明 


图 18.9 订购 列表 


商品 订购 和 提交 功能 的 后 台 主 要 包括 两 个 部 分 ， 一 部 分 是 业务 逻辑 控制 剖 
分 是 数据 库 访 问 部 分 。 


分， 另 一 部 


该 功能 完成 主要 由 BuyBookCmd.java、BookOrderSubmit.java 两 个 类 以 及 数据 库 操作 


代码 实现 。 


1. BuyBookCmd 的 实现 


该 类 主要 从 JSP 页 面 获取 用 户 订购 的 数据 ， 并 经 过 处 理 后 把 数据 放 到 SESSION 中 ， 


供 提 交 订 单 时 使 用 。BuyBookCmd 类 部 分 代码 如 下 : 


01 public void doGet (HttpServletRequest request, HttpServletResponse 
response) 
02 throws ServletException, IOException { 
03 // 得 到 相对 应 的 ID 
04 String bookid = request.getParameter ("bookid"); 
05 // 数量 
06 String count = request.getParameter ("count"); 
07 int i = 0; 
08 // 交 易 价格 
09 double tfp = (Double) request.getSession() .getAttribute 
(nSESSOTEP") 
10 // 物品 列表 
1 Map<Book, Integer> ptbooklist = (Map) request.getSession() 
二 2 getAttribute ("SESS BOOK"); 
13 
14 if (ptbooklist == null) { 
5 Pptbooklist = new HashMap<Book, Integer>(); 
16 } else { 
ph i = ptbooklist.size(); 
18 } 
9 
20 Ery { 
2 Book book = new BookDRAO () .getDetail (Long.parseLong (bookid)); 
2 book.setCount (count); 
1 book.setTotalprice (Double.valueOf (count) * book. 
getBookPrice()); 
24 tfp = tfp + Double.valueOf (count) * book.getBookPrice(); 


ss 
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之 5 // 存 入 Map 中 

26 ptbooklist.put (book, ++i); 

人 2 沁 } catch (SQLException e) { 

28 e.printStackTrace (); 

29 } 

30 request.setAttribute ("pctbook", ptbooklist); 

31 request.getSession() .setAttribute("SESS TFP", tfp); 

国光 request.getSession() .setAttribute ("SESS BOOK", ptbooklist); 

33 // 跳 转 传 值 

34 request.getRequestDispatcher ("/pctbook.jsp") .forward (request, 
response); 

35 1} 

其 中 : 

口 第 1 行 ， 重 写 doGet 方法， 获取 JSP 页 面 传递 数据 ， 并 对 数据 进行 处 理 。 

口 第 4 行 ， 获取 商品 ID。 

口 第 6 行 ， 获取 订购 数量 。 

口 第 9 行 ， 从 SESSION 中 获取 交易 价格 。 

口 第 11 行 ,从 SESSION 中 获取 订购 列表 。 

口 第 14 一 18 行 ， 判断 ptbooklist 是 否 为 室 ， 如 果 空 则 创建 一 个 实例 ， 假 如 不 为 空 ， 


要 获取 该 Map 对 象 的 size 属性 。 

第 21 行 ， 根 据 ID 获取 商品 〈 图 书 ) 对 象 。 
第 22 行 ， 保 存 订购 数量 。 

第 23 行 ， 保 存 订购 的 某 类 物品 价格 。 

第 24 行 ， 获 取 交 易 价格 。 

第 30 行 ， 把 值 放 入 request 范围 内 。 

第 31 行 ， 把 tfp 放 入 session 范围 内 。 

第 32 行 ， 把 ptbooklist 放 入 session 范围 内 。 


OOOOOODO 


DD 


. BookOrderSubmit 的 实现 


该 类 除了 从 JSP 页 面 获 取 用 户 订购 的 数据 外 ， 也 会 把 数据 进行 简单 的 业务 处 理 ， 完 成 
数据 封装 ， 调 用 访问 数据 库 代 码 ， 进 行 数据 存储 操作 。BookOrderSubmit 类 部 分 代码 如 下 : 


01 public void doGet (HttpServletRequest request, HttpServletResponse 


response) 

02 throws ServletException, IOException { 

03 

04 // 得 到 用 户 提交 的 数据 

05 String customerName = new String(request.getParameter 
("customerName") 

06 getBytes ("ISO-8859-1"), "GBK"); 

07 String address = new String(request.getParameter 
("customerAddress") 

08 .getBytes ("ISO-8859-1"), "GBK"); 

09 String telephone = request.getParameter ("customerTelephone"); 

10 String notice = new String(request.getParameter ("notice"). 
getBytes ( 

六 "T908059=-17)7 "GBE")s 

ba String totalPrice = request.getParameter("allPrice"); 

13 String sessuser = (String) request.getSession() .getRttribute( 

14 "SESS_USER"); 


“388。 
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44 } 


其 中 : 


DOOOOO 
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Map<Book, Integer> pcntbooklist = (Map) request.getSession() 
.getAttribute ("SESS BOOK"); 
String bookname = "订购 "; 
Iterator it = pcntbooklist.entrySet().iterator(); 
while (it-hasNext()) { // 遍 历 iterator 对 象 
Map .Entry m = (Map.Entry) it.next() 


bookname = bookname + ";" + ((Book) m.getKey()).getBookName() 
+ "xn+ ((Book) m.getKey()) .getCount () + " 份 "; 
// 获 取 数 据 连接 字符 串 
} 


// 封装 BEAN 

BookOrder bookoder = new BookOrder (); 
bookoder.setCustomerName (customerName); 
bookoder.setCustomerAddress (address); 
bookoder.setCustomerTelephone (telephone); 
bookoder.setNotic (notice); 

bookoder.setTotalPrice (Double.parseDouble (totalPrice)); 
bookoder.setSessuser (sessuser); 

bookoder .setMyBookId (String.valueOf (bookname) ); 


// 订购 
try { 

new DoBookOrder() .insertBookOrderInfo (bookoder); 
} catch (SQLException e) { 

e.printStackTrace (); 


} 
// 跳 转 


response.sendRedirect ("over.jsp"); 


第 5 一 12 行 ， 获 取 页 面 数 据 ， 并 保证 汉字 不 为 乱码 。 

第 15 行 ， 从 SESSION 中 获取 订购 商品 。 

第 19 行 ， 遍 历 iterator 对 象 。 

第 22 行 ， 从 遍历 对 象 中 获取 数据 ， 连 接 成 符合 要 求 的 字符 串 。 
第 27 一 34 行 ， 把 所 有 数据 封装 到 BookOrder 对 象 中 。 

第 38 行 ， 调 用 DoBookOrder 类 的 方法 ， 把 数据 存 入 数据 库 中 。 


到 这 里 系统 的 主要 部 分 都 介绍 得 差不多 了 ， 其 他 功能 点 由 于 篇 幅 关系 ， 这 里 不 再 详细 
介绍 ， 读 者 可 以 参考 前 面 介绍 的 功能 点 对 照 代码 自己 理解 。 


18.4 本 章 小 结 


本 章 中 介绍 了 B/S 结构 ,同时 介绍 了 B/S 结构 的 优势 。 浏 览 器 /服务 器 模式 需要 服务 器 


作 容 器 ， 常 
为 服务 器 。 


除了 介绍 B/S 结构 编程 ， 书 中 也 使 用 了 一 个 简单 的 在 线 订 购 系统 详细 介绍 了 如 何 利用 
Java 连接 数据 库 。Java 连接 数据 库 ， 可 以 使 用 JDBC， 也 可 以 使 用 JNDI， 其 中 JNDI 更 符 


的 服务 器 有 Tomcat、Jboss 和 WebLogic 等 。 本 书 实例 就 是 使 用 了 Tomcat 作 


合 企业 级 的 要 求 。 在 系统 中 这 两 种 方式 都 有 使 用 , 读者 可 以 利用 这 个 简单 的 在 线 购物 系统 ， 
完善 自己 的 相关 知识 。 


ss 


